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从 入 [门人 到 精通 


6 小 时 语音 视频 讲解 
上 加 实例 资源 库 加 模块 资源 库 ”加 项 目 资源 库 
甸 加 面试 资源 库 “ 回 测 试题 库 系 统 四 PPT 电 子 课件 


组 循序 渐进 ， 实 战 讲述 
基础 知识 加 核心 技术 串 高 级 应 用 品 项 目 实战 
219 个 学 习 实例 ，46 个 练习 实例 ，2 个 项 目 案例 
@ 海量 资源 ， 可 查 可 练 


除 本 书 配套 的 6 小 时 视频 讲解 外 ， 根 据 学 习 顺 序 ， 光 盘 还 额 
外 配备 如 下 海量 开发 资源 库 


实例 资源 库 (808 个 实例 ) 吃 模块 资源 库 (15 个 典型 模块 ) 吃 
项 目 资源 库 (15 个 项 目 案例 ) 测试 题库 系统 (626 道 测试 题 ) 
只 面试 资源 库 (342 个 面试 真题 ) 

食 在 线 解 答 ， 高 效 学 习 

QQ: 400 675 1066( 可 容纳 10 万 人 在 线 ) 
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MySQL 从 入 门 到 精通 


明日 科技 ”编著 


内 容 简 介 


《MySQL 从 入 门 到 精通 》 从 初学 者 角度 出 发 ， 通 过 通俗 易 懂 的 语言 以 及 丰富 多 彩 的 实例 ， 详 细 介绍 了 MySQL 开 
发 应 该 掌握 的 各 方面 技术 。 全 书 共 分 为 4 篇 23 章 ， 包 括 数据 库 基础 ， 初 识 MYSQL， 使 用 MySQL 图 形 化 管理 工具 ， 数 
据 库 操作 ， 存 储 引擎 及 数据 类 型 ， 操 作 数据 表 ，MYySQL 基础 ， 表 数据 的 增 、 删 、 改 操作 ， 数 据 查 询 ， 常 用 函数 ， 索 引 ， 
视图 ， 数 据 完 整 性 约束 ， 存 储 过 程 与 存储 函数 ， 触 发 器 ， 事 务 的 应 用 ， 事 件 ， 备 份 与 恢复 ，MySQL 性 能 优化 ， 权 限 管 
理 及 安全 控制 ，PHP 管理 MySQL 数据 库 中 的 数据 ，Apache+PHP+MySQL 实现 网 上 社区 ，Struts 2+Spring+Hibernate+ 
MySQL 实现 网 络 商城 等 内 容 。 所 有 知识 都 结合 具体 实例 进行 介绍 ， 涉 及 的 程序 代码 也 给 出 了 详细 的 注释 ， 可 以 使 读者 
轻松 领会 MySQL 的 精髓 ， 快 速 提高 开发 技能 。 

另外 ， 本 书 除了 纸 质 内 容 之 外 ， 配 套 光盘 中 还 给 出 了 海量 开发 资源 库 ， 主 要 内 容 如 下 : 


回 语音 视频 讲解 : 总 时 长 6 小 时 ， 共 68 段 实例 资源 库 : 808 个 实例 及 源码 详细 分 析 

回 模块 资源 库 : 15 个 经 典 模块 开发 过 程 完整 展现 项 目 案例 资源 库 : 15 个 企业 项 目 开发 过 程 完 整 展 现 
回 测试 题库 系统 : 626 道 能 力 测试 题目 面试 资源 库 : 342 个 企业 面试 真题 

PPT 电子 教案 


本 书 内 容 详尽 ， 实 例 丰 富 ， 非 常 适合 作为 编程 初学 者 的 学 习 用 书 ， 也 适合 作为 开发 人 员 的 查阅 、 参 考 资料 。 


本 书 封面 贴 有 清华 大 学 出 版 社 防伪 标签 ， 无 标签 者 不 得 销售 。 
版 权 所 有 ， 侵 权 必 究 。 侵 权 举 报 电话 : 010-62782989 13701121933 


图 书 在 版 编目 (CIP) 数据 

MySQL 从 入 门 到 精通 /明日 科技 编著 .一 北京 : 清华 大 学 出 版 社 ，2017 
(软件 开发 视频 大 讲堂 

ISBN 978-7-302-45799-2 


I DM… I @ 明 … ”II DSQL 语言 -程序 设计 IV. TP311.132.3 





中 国 版 本 图 书馆 CIP 数据 核 字 (2016) 第 290847 号 











责任 编辑 : 赵 洛 育 
封面 设计 : 刘 洪 利 
版 式 设计 : 李 会 影 
责任 校对 : 何 士 如 
责任 印 制 : 
出 版 发 行 : 清华 大 学 出 版 社 
网 http://www.tup.com.cn, http://www.wqbook.com 
地 : 北京 清华 大 学 学 研 大 厦 A 座 邮 编 : 100084 
社 总 加 010-62770175 邮  ” 购 : 010-62786544 


投稿 与 读者 服务 : 010-62776969，c-service@tup.tsinghua.edu.cn 
质量 反馈 : 010-62772015，zhiliang@tup.tsinghua.edu.cn 


印刷 者 : 

装订 者 : 

经 ” 销 : 全 国 新 华 书店 

开 本 : 203mmX260mm 印张 : 325 字 数 : 889 千 字 
( 附 DVD 视频 光盘 1 张 ) 

版 ”次 : 2017 年 9 月 第 1 版 印 ”次 : 2017 年 9 月 第 1 次 印刷 

印 ” 数 : 1 一 5000 

定 ” 价 : 79.80 元 





产品 编号 : 058862-01 


如 何 使 用 本 书 开发 资源 库 


在 学 习 《MySQL 从 入 门 到 精通 》 一 书 时 ， 随 书 附 配 光盘 提供 了 “PHP 开发 资源 库 ” 系 统 ， 可 以 帮 
助 读者 快速 提升 编程 水 平和 解决 实际 问题 的 能 力 。《MySQL 从 入 门 到 精通 》 和 PHP 开发 资源 库 配合 
学 习 流程 如 图 1 所 示 。 


MySQL 从 入 


门 到 精通 | 下 


图 1 图 书 与 开发 资源 库 配 合 学 习 流程 图 


打开 光盘 的 “开发 资源 库 ” 文 件 夹 ， 运 行 PHP 开发 资源 库 .exe 程序 ， 即 可 进入 “PHP 开发 资源 
库 ” 系 统 ， 主 界面 如 图 2 所 示 。 


MA ED 和 










出 强 


吏 和 门 川 外 营 


学 习 ， 其 他 部 分 没有 开放 ， 不 能 免费 | 
陆 www. mrbccd. com 了 解 或 购买 。 


























图 2 PHP 开发 资源 库 主 界面 
在 学 习 某 一 章节 时 ， 可 以 配合 实例 资源 库 的 相应 章节 ， 利 用 实例 资源 库 提供 的 大 量 热点 实例 和 关 
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键 实例 巩固 所 学 编程 技能 ， 提 高 编程 兴趣 和 自信 心 ; 也 可 以 配合 能 力 测试 题库 的 对 应 章节 进行 测试 ， 
检验 学 习 成 果 。 具 体 流程 如 图 3 所 示 。 
能 力 ， 实 时 检 


MySQL 从 入 门 到 精通 
能 力 测试 题库 测 学 习 成 果 


图 3 使 用 实例 资源 库 和 能 力 测试 题库 


对 于 数学 逻辑 能 力 和 英语 基础 较为 薄弱 的 读者 ， 或 者 想 了 解 个 人 数学 逻辑 思维 能 力 和 编程 英语 基 
础 的 用 户 ， 本 书 提供 了 数学 及 逻辑 思维 能 力 测试 和 编程 英语 能 力 测试 供 练习 和 测试 ， 如 图 4 所 示 。 

肚 编程 基础 能 力 测试 

日 六 数学 及 加 思维 能 力 W 坛 志 一 一 | 练习 和 检测 数学 及 计 辑 思维 能 力 

[a 基本 测试 

于 

Te 奈 林 和 个 天 尖 训 册 全 


租 英语 基础 能 力 测试 


图 4 数学 及 逻辑 思维 能 力 测试 和 编程 英语 能 力 测试 目录 


当 本 书 学 习 完成 时 ， 可 以 配合 模块 资源 库 和 项 目 资源 库 的 30 个 模块 和 项 目 ， 全 面 提升 个 人 综合 编 
程 技能 和 解决 实际 开发 问题 的 能 力 ， 为 成 为 PHP 软件 开发 工程 师 打 下 坚实 基础 。 具体 模块 和 项 目 目录 
如 图 5 所 示 。 










实例 资源 库 快速 提升 编程 

































多 | 九 九 度 供求 信息 网 
多 BCTY365 网 上 社区 
多 办公 自 动 化 管理 系统 


多 注册 及 登录 验证 模块 





由 名 网 站 计数 分 析 醒 所 渔 电子 商务 平台 网 

由 - 侈 留言 本 模块 渔网 络 在 线 兰 试 系统 

沟 搜索 引擎 模块 田 -多 | 物流 配送 信息 网 

多 投票 系统 模块 由 - 钨 字 校 图 书馆 管理 系统 

购物 车 模块 由 - 钨 博客 管理 系统 

移 在 线 支付 模块 由 - 钨 企业 快 信 系统 

钨 聊天 室 模块 田 钢 online 影 视 365 网 

多 | 论坛 模块 田 - 侈 ] 明日 知道 网 站 

多 | 许愿 墙 模块 多 电子 商务 网 站 

多 博客 模块 多 新 闻 发 布 系统 一 毕业 设计 的 首选 
网 络 象棋 游戏 模块 由 - 侈 明日 企业 网 一 企业 网 站 开发 模板 


明日 知道 一 程序 员 的 十 万 个 为 什么 
图 5 模块 资源 库 和 项 目 资源 库 目 录 
万 事 俱 备 ， 该 到 软件 开发 的 主 战场 上 接受 洗礼 了 。 面 试 资源 库 提供 了 大 量 国 内 外 软件 企业 的 常见 
面试 真题 ， 同 时 还 提供 了 程序 员 职 业 规 划 、 程 序 员 面 试 技巧 、 企 业 面 试 真题 汇编 和 虚拟 面试 系统 等 精 
彩 内 容 ， 是 程序 员 求 职 面试 的 绝 佳 指南 。 面 试 资源 库 的 具体 内 容 如 图 6 所 示 。 


如 何 使 用 本 书 开发 资源 库 


已 - 侦 上 
由 -由 第 ! 部 分 。 PHE 程序 员 职 业 规 划 
1- 般 第 2 部 分 。 PIP 程序 员 面试 技巧 


EE 

由 -和 贡 第 3 部 分 。 FIP 常见 面试 题 

所 -入 第 4 部 分 。 PP 企业 面试 真是 汇编 
大 企业 面 工 真题 汇编 (一 ) 
地 企业 面 荆 真题 编 ( 二 ) 
办 企业 面 工 丰 题 i 编 (三 ? 
侈 企业 面 计 丰 显 汇编 ( 四 ) 

由 - 贡 第 5 部 分 虚拟 面试 系统 

田阳 编程 人 生 


图 6 面试 资源 库 的 具体 内 容 


如 果 您 在 使 用 PHP 开发 资源 库 时 遇 到 问题 ， 可 加 我 们 的 QQ: 4006751066( 可 容纳 10 万 人 )， 我 
们 将 竭诚 为 您 服务 。 

















了 


和 
出 


MySQL 数据 库 是 世界 上 最 流行 的 数据 库 之 一 。 全 球 最 大 的 网 络 搜索 引擎 公司 Google 使 用 的 数据 
库 就 是 MySQL， 并 且 国内 的 很 多 大 型 网 络 公司 也 选择 MySQL 数据 库 ， 如 百度 、 网 易 和 新 浪 等 。 据 统 
计 ， 世 界 上 一 流 的 互联 网 公司 中 ， 排 名 前 20 位 的 有 80% 是 MySQL 的 忠实 用 户 。 目 前 ，MySQL 已 经 
被 列 为 全 国 计 算 机 等 级 考试 二 级 的 考试 科目 。 



































本 书 内 容 
本 书 提供 了 从 入 门 到 编程 高 手 所 必 备 的 各 类 知识 ， 共 分 为 4 篇 ， 大 体 结构 如 下 所 示 。 
第 1 篇 : 基础 知识 
快速 浏览 本 章 内 容 
知识 讲解 















































图 示 
第 2 篇 ; 核心 技术 实例 、 视 频 










































































入 | 注意、 说明、 技巧 
小 结 
高 手 
实践 与 练习 
第 3 篇 : 高 级 应 用 
快速 浏览 本 章 内 容 、 项 目 开 发 全 
人 过 程 、 图 示 、 视 频 等 














第 1 篇 ， 基础 知识 。 本 篇 通过 对 数据 库 基础 、 初 识 MySQL、 使 用 MySQL 图 形 化 管理 工具 、 数 据 
库 操作 、 存 储 引 擎 及 数据 类 型 和 操作 数据 表 等 内 容 的 介绍 ， 并 结合 大 量 的 图 示 、 举 例 、 视 频 等 使 读者 
快速 掌握 MySQL， 并 为 以 后 的 知识 葛 定 坚实 的 基础 。 

第 2 篇 : 核心 技术 。 本 篇 介绍 MySQL 基础 ， 表 数据 的 增 、 删 、 改 操作 ， 数 据 查 询 ， 常 用 函数 ， 
索引 ,视图 等 内 容 。 学 习 完 这 一 部 分 ， 能够 了 解 和 熟悉 MySQL 及 常用 的 函数 ， 使 用 SQL 操作 MySQL 
数据 库 中 的 视图 ， 掌 握 SQL 查询 、 子 查询 、 嵌 套 查询 、 连 接 查 询 的 用 法 等 。 

第 3 篇 : 高 级 应 用 。 本 篇 介绍 数据 完整 性 约束 、 存 储 过 程 与 存储 函数 、 触 发 器 、 事 务 的 应 用 、 事 
件 、 备 份 与 恢复 、MySQL 性 能 优化 、 权 限 管理 及 安全 控制 、PHP 管理 MySQL 数据 库 中 的 数据 等 内 容 。 
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学 习 完 这 一 部 分 ， 能 够 掌握 如 何 进行 数据 的 导入 与 导出 操作 ， 以 及 使 用 存储 过 程 、 触 发 器 、 事 务 、 事 
件 等 。 通 过 这 些 内 容 不 仅 可 以 优化 查询 ， 还 可 以 提高 数据 访问 速度 ， 更 好 地 维护 MySQL 的 权限 及 其 
安全 。 另 外 ， 还 介绍 了 应 用 PHP 管理 MySQL 数据 库 中 的 数据 ， 对 于 想 要 使 用 PHP 开发 的 读者 非常 
实用 。 

第 4 篇 : 项 目 实战 。 本 篇 分 别 使 用 PHP 和 Java 两 种 语言 ， 结 合 MySQL 实现 了 两 个 大 型 的 、 完 整 
的 管理 系统 ， 通 过 这 两 个 项 目 ， 运 用 软件 工程 的 设计 思想 ， 帮 助 读者 学 习 如 何 进 行 软件 项 目的 实践 
开发 。 书 中 按照 编写 系统 分 析 一 系统 设计 一 数据 库 与 数据 表 设计 一 公共 模块 设计 一 创建 项 目 一 实现 
项 目 一 项 目 总 结 的 过 程 进行 介绍 ， 带 领 读 者 一 步 一 步 亲 身体 验 开 发 项 目的 全 过 程 。 


本 书 特点 


号 加 由浅 入 深 , 循序 渐进 : 本 书 以 初 、 中 级 程序 员 为 对 象 ， 先 从 MySQL 基础 学 起 , 再 学 习 MySQL+- -- 
的 核心 技术 ,然后 学 习 MySQL 的 高 级 应 用 ,最 后 学 习 分 别 使 用 PHP 和 Java 等 语言 结合 MySQL 
开发 完整 项 目 。 讲 解 过 程 中 步骤 详尽 ， 版 式 新 颖 ， 在 操作 的 内 容 图 片上 以 “@8@@…” 编 号 + 
内 容 的 方式 进行 标注 ， 让 读者 在 阅读 中 一 目 了 然 ， 从 而 快速 把 握 书 中 内 容 。 

号 加 语音 视频 ,讲解 详尽 ; 书 中 每 一 章节 均 提 供 声 图 并 茂 的 教学 视频 , 读者 可 以 在 光盘 中 找到 相应 
章节 的 视频 。 这 些 视频 能 够 引导 初学 者 快速 入 门 ， 感 受 编程 的 快乐 和 成 就 感 ， 增 强 进一步 学 习 
的 信心 ， 从 而 快速 成 为 编程 高 手 。 

号 加 实例 典型 ， 轻 松 易学 : 通过 例子 学 习 是 最 好 的 学 习 方 式 ， 本 书 通过 一 个 知识 点 、 一 个 例子 、 一 
个 结果 、 一 段 评析 和 一 个 综合 应 用 的 模式 ， 透 彻 、 详 尽 地 讲述 了 实际 开发 中 所 需 的 各 类 知识 。 

号 加 精彩 栏目 ， 贴 心 提醒 ， 本 书 根据 需要 在 各 章 使 用 了 很 多 “注意 ”“ 说 明 ” 和 “技巧 ”等 小 栏 
目 ， 让 读者 可 以 在 学 习 过 程 中 更 轻松 地 理解 相关 知识 点 及 概念 ， 并 轻松 地 掌握 个 别 技术 的 应 
用 技巧 。 

号 回应 用 实践 ， 随 时 练习 ， 书 中 几乎 每 章 都 担 供 了 “实践 与 练习 ”， 读 者 能 够 通过 对 问题 的 解答 重 
新 回顾 、 热 悉 所 学 的 知识 ， 举 一 反 三， 为 进一步 学 习 做 好 充分 的 准备 。 


读者 对 象 


回 ”初学 编程 的 自学 者 编程 爱好 者 

大 中 专 院 校 的 老师 和 学 生 相关 培训 机 构 的 老师 和 学 员 
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读者 服务 
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冯 春 龙 、 宋 万 勇 、 李 文 欣 、 王 东 东 、 柳 琳 、 王 盛 夸 、 徐 明明 、 杨 柳 、 赵 宁 、 王 佳 雪 、 于 国良 、 李 磊 、 
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免 ， 敬 请 广大 读者 批评 指正 。 我 们 的 服务 邮箱 是 tmoonbook@sina.com， 了 h_press@263.net。 读 者 在 阅读 
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感谢 您 购买 本 书 ， 希 望 本 书 能 成 为 您 编程 路 上 的 领航 者 。 

“ 零 门 槛 ”编程 ， 一 切 错 有 可 能 。 

祝 读书 快乐 ! 
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删除 FTP 服务 器 中 指定 的 文件 

在 FTP 服务 器 中 建立 指定 的 目录 

获取 FTP 服务 器 中 指定 目录 下 的 文件 列表 
PHP 伪 静 态 的 实现 

MySQL 数据 库 双 机 热 备份 

关 PHP 国际 化 


国 





Fckeditor 文本 编辑 器 的 应 用 
避免 截取 中 文字 符 串 时 乱码 
PHP 动态 生成 静态 页 面 

在 线 支付 一 工商 银行 

在 线 支付 一 支付 宝 支付 
PHP 与 WebService 交互 





设计 GB2312 编码 格式 的 网 页 
设计 GBK 编码 格式 的 网 页 
设计 UTF-8 编码 格式 的 网 页 
PHP 的 国际 化 
数据 库 连接 类 
数据 库 管理 类 
数据 库 分 页 类 

Smarty 模板 引擎 配置 类 
字符 串 处 理 类 

网 站 的 头 文件 设计 

网 站 的 尾 文件 设计 
首页 广告 设计 

用 户 注册 
分 步 用户 注 册 

用 户 登 录 

用 户 中 心 

用 户 安全 退出 
我 的 订单 

找 回 密码 

图 书 导航 

图 书 分 类 

特别 图 书 

图 书 试 读 

图 书 详细 信息 展示 

新 闻 公告 

一 般 搜索 

高 级 搜索 

常用 搜索 

购物 车 类 
购物 车 功能 实现 
填写 收 货 人 信息 

确认 订购 信息 
支付 宝 在 线 支付 
工行 在 线 支 付 
管理 员 登 录 









天 | 后 台 管理 系统 主页 设计 
系统 信息 设置 

更 改 管理 员 密码 
书 大 类 管理 
书 小 类 管理 

出 版 社 分 类 管理 








天 | 新 闻 公告 管理 


MySQL 从 入 门 到 精通 


简易 留言 本 

带 留言 分 类 的 留言 本 
具有 版 主 回复 的 留言 本 
数据 库 形式 的 聊天 室 
聊天 室 中 私 聊 的 实现 
查看 主题 信息 
发 布 主 题 信息 
回复 主题 信息 
删除 主题 及 回复 信息 
博客 用 户 图 片 管理 
博客 文章 评论 管理 





第 2 大 部 分 “模块 资源 库 


(15 个 经 典 模块 ， 光 盘 路 径 : 


模块 1 注册 及 登录 验证 模块 
口 国 注 册 及 登录 验证 模块 概述 
于 | 用 户 注册 流程 
鲁 | 用 户 登 录 流程 
国 找 回 密码 流程 
日 国 热 点 关键 技术 
] 防 SQL 注入 技术 
Aiax 技术 实现 无 刷新 验证 
验证 码 技术 
鲁 E-mail 激活 技术 
应 用 键盘 响应 事件 验证 信息 是 否 合法 
应 用 Cookie 技术 实现 自动 登录 








验证 码 的 实现 与 刷新 
国 找 回 密码 的 实现 

日 国 程序 调试 

国 程序 调试 


模块 2 上 传 下 载 模块 
局 国 上 传 下 载 模块 








XX 


开发 资源 库 /模块 资源 库 ) 


上 传 下 载 模块 概述 
热点 关键 技术 

5 国 实 现 过 程 
数据 库 设计 
文件 上 传 功能 的 实现 《包括 多 文件 上 传 ) 
文件 下 载 的 实现 

5 国 程 序 调试 
程序 调试 

模块 3 分 页 显示 模块 

5 国 分 页 显示 模块 
分 页 显示 模块 概述 
热点 关键 技术 

是 国 分 页 类 模块 
Smarty 模板 的 安装 和 配置 
ADODB 的 配置 和 连接 
分 页 类 模块 的 页 面 设 计 
分 页 类 模块 的 程序 开发 

5 国 分 页 显示 模块 的 实现 
PHP 超 长 文本 分 页 功能 的 实现 
Ajax 无 刷新 分 页 功能 的 实现 
PHP 跳 转 分 页 功能 的 实现 
PHP 上 下 分 页 功能 的 实现 


光盘 “开发 资源 库 ” 目 录 


启 国 程 序 调试 
全 | 程序 调试 


模块 4 网 站 计数 分 析 模 块 
口 国 网 站 计数 分 析 模 块 
国 网 站 计数 分 析 模块 概述 
国 热点 关键 技术 
日 国 计 数 器 

国 简单 数字 计数 器 
数字 计数 器 
GD2 图 形 计数 器 













Session 无 刷新 计数 器 
口 国 网 站 计数 器 分 析 模块 

国 计数 器 功能 的 实现 
网 站 访问 量 统计 分 析 
国 应 用 GD2 函数 动态 创建 折线 图 
日 国 程序 调试 
国 程序 调试 
模块 5 留言 本 模块 
口 国 留 言 本 模块 概述 

国 留言 本 概述 
留言 本 的 功能 结构 
留言 本 系统 流程 
日 国 热点 关键 技术 

国 什么 是 敏感 词 

国 过 小 敏感 词 

国 | 添加 敏感 词 到 文本 文件 中 

鲁 ] 读 取 文 本 文件 中 的 敏感 词 

避 ] 验证 码 在 当前 页 验证 

国 实现 复 选 框 的 全 选 和 反选 
口 国 实现 过 程 

国 MysQL 数据 库 设计 

国 定义 数据 库 访问 类 
签 写 留言 及 过 滤 敏感 词 的 实现 
分 页 查看 留言 及 版 主 信息 回 显 的 实现 
检索 留言 及 回复 信息 的 实现 
版 主 回复 留言 功能 的 实现 
国 ; 批量 删除 留言 及 回复 信息 的 实现 
国 版 主 悄悄 话 管理 功能 的 实现 




















模块 6 搜索 引擎 模块 
口 国 搜索 引擎 模块 
搜索 引擎 模块 概述 
热点 关键 技术 
口 国 实现 过 程 
RMM 分 词 查询 的 实现 
在 查询 结果 中 二 次 搜索 功能 的 实现 
高 级 搜索 功能 的 实现 
知名 站 点 互联 网 查询 功能 的 实现 
5 国 程 序 调试 
程序 调试 
模块 7 投票 系统 模块 
5 国 投票 系统 模块 
投票 系统 模块 概述 
投票 关键 技术 
口 国 简易 投票 系统 
动态 生成 投票 主题 
动态 添加 投票 选项 内 容 
投票 主题 内 容 管理 
投票 功能 的 实现 
= 国 复杂 投票 系统 
复杂 投票 系统 
程序 调试 
模块 8 购物 车 模块 
日 国 购 物 车 模块 
购物 车 功能 概述 
购物 车 操作 流程 
Smarty 模板 的 安装 配置 
Smarty 模板 动静 分 离 
Session 购物 车 的 创建 
通过 数组 函数 判断 购物 车 是 否 存在 指定 商品 
验证 输入 商品 数量 的 值 是 否 有 效 
5 国 实现 过 程 与 错误 处 理 
商品 展示 功能 的 实现 
添加 商品 功能 的 实现 
删除 购物 车 中 商品 功能 的 实现 
更 改 购物 车 中 商品 数量 功能 的 实现 
统计 购物 车 中 商品 金额 功能 的 实现 
清空 购物 车 中 商品 功能 的 实现 


MySQL 从 入 门 到 精通 


国 填写 订单 信息 功能 的 实现 
生成 订单 功能 的 实现 
国 错误 处 理 
模块 9 ”在线 支 付 模块 
口 国 在 线 支付 模块 
国 | 在 线 支付 模块 概述 
镜 收 货 人 信息 验证 
国 动态 生成 订单 号 
WebBrowser 打印 
工行 支付 
国 支付 宝 支付 
口 国 实现 过 程 与 程序 调试 
国 商品 展示 功能 的 实现 
镜 购物 车 功能 的 实现 
国 订单 打印 和 预览 功能 的 实现 
提交 订单 功能 的 实现 
国 订单 查询 功能 的 实现 
什 程序 调试 















模块 10 ”聊天 室 模块 
口 国难 天 宝 模 块 
聊天 室 模块 概述 
应 用 框架 布局 聊天 室 主页 面 
文件 操作 技术 
定时 刷新 技术 删除 不 发 言 用 户 
滚屏 显示 时 滚动 条 定位 技术 
屏蔽 刷新 技术 
监控 客户 端 浏览 器 
5 国 实现 过 程 与 程序 调试 
聊天 室 模块 实现 过 程 
配置 文件 (configphp) 
公共 函数 文件 
用 户 登 录 验 证 
发 言 功能 的 实现 
用 户 列表 功能 的 实现 
公共 聊天 功能 和 私 聊 功能 的 实现 
自动 删除 掉 线 用 户 功能 的 实现 
程序 调试 





第 3 大 部 分 “项目 资源 库 


《15 个 企业 开发 项 目 ， 光 盘 路 径 : 开发 资源 库 /项 目 资源 库 ) 


项 目 1 九 九 度 供求 信息 网 


项 目 2 BCTY365 网 上 社区 
日 国 开 发 背景 和 系统 分 析 


仁 ] 开发 背景 





国 可 行 性 分 析 

国 编写 项 目 计划 书 
口 国 系统 设计 

国 系统 目标 

国 系统 功能 结构 

国 系统 预览 

国 开发 环境 

下 文件 顽 组 织 结构 





5 国 在 Linux 操作 系统 下 搭建 PHP 开发 环境 
Linux 下 Apache 的 安装 配置 
Linux 下 MySQL 的 安装 配置 
Linux 下 PHP 的 安装 配置 

5 国 数据 库 设计 
数据 库 分 析 
数据 库 概念 设计 
创建 数据 库 及 数据 表 

习 国 公 共 模 块 设计 和 前 台 首页 设计 
数据 库 连 接 文件 
将 文本 中 的 字符 转换 为 HTML 标识 符 
前 台 首页 概述 
前 台 首 页 技术 分 析 
前 台 首页 的 实现 过 程 

口 国 注 册 模块 设计 

口 国 技术 支持 模块 设计 





国 ; 技术 支持 模块 概述 
技术 支持 模块 技术 分 析 
国 | 常见 问题 的 实现 过 程 
国 客户 反馈 的 实现 过 程 
国 单元 测试 
口 国 在 线 订购 模块 设计 
鲁 | 在 线 订购 模块 概述 
鲁 | 在 线 订购 模块 技术 分 析 
国 ; 购物 车 的 实现 过 程 
国 | 商品 订单 的 实现 过 程 
国 单元 测试 
日 国 社区 论坛 模块 设计 
国 社区 论坛 模块 概述 












论坛 帖子 发 布 的 实现 过 程 
论坛 帖子 回复 的 实现 过 程 
国 单元 测试 
口 国 后 台 首页 设计 
口 国 编程 词典 管理 模块 设计 
编程 词典 管理 模块 概述 
编程 词典 管理 模块 技术 分 析 
国 。 添加 编程 词典 的 实现 过 程 
国 编辑 编程 词典 的 实现 过 程 
日 国 软 件 升级 管理 模块 设计 
国 软件 升级 管理 模块 概述 
软件 升级 管理 模块 技术 分 析 
国 软件 升级 包 上 传 的 实现 过 程 
国 软件 升级 包 删 除 的 实现 过 程 
口 国 在 Linux 系统 下 发 布 网 站 
口 国 开发 技巧 与 难点 分 析 
国 管理 员 权限 的 设置 
国 帖子 置顶 的 设置 
日 国 在 线 支付 技术 专题 


项 目 3 ”办公 自动 化 管理 系统 
口 国 开 发 背景 和 需求 分 析 
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系统 目标 
系统 功能 结构 
系统 功能 预览 
系统 流程 图 
开发 环境 
文件 夹 组 织 结构 

5 国 数 据 库 设计 和 公共 模块 设计 
数据 库 分 析 
数据 库 概念 设计 
数据 库 物 理 结构 设计 
JavaScript 脚本 
自 定义 函数 

局 国 前 台 首页 设计 与 人 事 消息 模块 设计 
前 台 首 页 概述 
前 台 首页 技术 分 析 
前 台 首 页 的 实现 过 程 
人 事 消息 模块 概述 
人 事 消息 模块 技术 分 析 
消息 管理 的 实现 过 程 
意见 箱 的 实现 过 程 

日 国 考 勤 管理 模块 设计 
考勤 管理 模块 概述 
考勤 管理 模块 技术 分 析 
上 下 班 登记 的 实现 过 程 
设置 时 间 的 实现 过 程 

口 国 后 台 首页 设计 与 部 门 管理 模块 设计 
后 台 首页 概述 
后 台 首页 技术 分 析 
后 台 首 页 的 实现 过 程 
部 门 管理 模块 概述 
部 门 管理 模块 技术 分 析 
部 门 查看 的 实现 过 程 
部 门 添加 的 实现 过 程 
单元 测试 

5 国 系统 管理 模块 设计 
系统 管理 模块 概述 
系统 管理 模块 技术 分 析 
系统 日 志 的 实现 过 程 
数据 备份 的 实现 过 程 

日 国 开发 技巧 、 难 点 分 析 与 MySQL 数据 备份 专题 
使 用 JavaScript 关联 多 选 列表 框 


伍 ] 用 户 组 设置 
伍 ] MySQL 数据 备份 专题 


项 目 4 电子 商务 平台 
日 国 开发 背景 和 系统 分 析 


会 ] 开发 背景 
伍 ] 系统 分 析 
秸 ] 需求 分 析 
会] 可 行 性 分 析 





口 国 系统 设计 








数据 库 分 析 


数据 库 概念 设计 
天 | 创建 数据 库 及 数据 表 
全 ] 数据 库 连 接 文件 
舍 ] CSS 样式 表 文 件 


国 前 台 首页 设计 
国 商品 展示 模块 设计 


国 商品 展示 模块 概述 

国 商品 展示 模块 技术 分 析 

国 商品 分 类 展示 的 实现 过 程 

国 ) 最 新 商品 展示 的 实现 过 程 

国 ) 查看 商品 详细 信息 的 实现 过 程 
国 单元 测试 

购物 车 模块 设计 

国 | 网 站 购物 车 概述 

国 网 站 购物 车 技术 分 析 

国 ; 添加 至 购物 车 的 实现 过 程 

国 ) 查看 购物 车 的 实现 过 程 

国 从 购物 车 中 移 去 指定 商品 的 实现 过 程 
修改 商品 购买 数量 的 实现 过 程 
国 清空 购物 车 的 实现 过 程 

国 ; 收银 台 结 账 的 实现 过 程 

生成 商品 订单 的 实现 过 程 
国 单元 测试 











MySQL 从 入 门 到 精通 


5 国 后 台 首页 设计 与 客户 订单 信息 管理 模块 设计 


后 台 首 页 概述 

后 台 首 页 技术 分 析 

后 台 首 页 的 实现 过 程 

客户 订单 信息 管理 模块 概述 
客户 订单 信息 管理 模块 技术 分 析 
查看 客户 订单 信息 的 实现 过 程 
执行 客户 订单 信息 的 实现 过 程 
打印 客户 订单 信息 的 实现 过 程 
查找 客户 订单 信息 的 实现 过 程 


5 国 开发 技巧 、 难 点 分 析 与 加 密 技 术 专 题 
防止 非法 用 户 绕 过 系统 登录 直接 进入 系统 


检测 用 户 名 是 否 已 经 注册 
用 户 安全 退出 

URL 编码 加 密 技术 
base64 编码 加 密 技术 
crypt0 加 密 技术 
md50 加 密 技术 


项 目 5 网 络 在 线 考试 系统 
5 国 开 发 背景 和 系统 分 析 
开发 背景 
需求 分 析 
可 行 性 分 析 
归 国 系统 设计 
系统 目标 
系统 功能 结构 
系统 流程 图 
系统 预览 
开发 环境 
文件 夹 组 织 结构 
日 国 数 据 库 设计 与 前 台 首 页 设计 
数据 库 分 析 
数据 库 概念 设计 
数据 库 物 理 结构 
前 台 首页 概述 
前 台 首页 技术 分 析 
前 台 首页 的 实现 过 程 
归 国 考生 信息 模块 设计 
考生 信息 模块 概述 
考生 信息 模块 的 技术 分 析 
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日 


国 考生 注册 的 实现 过 程 
国 单元 测试 
国 在 线 考试 模块 设计 
国 ; 在 线 考试 模块 的 概述 
国 ) 在 线 考试 模块 的 技术 分 析 
国 应 用 Ajax 在线 答题 的 实现 过 程 
国 分 数 统计 和 成 绩 保存 的 实现 过 程 
单元 测试 
国 后 台 首页 设计 
国 考题 信息 管理 模块 设计 
鲁 | 考题 信息 管理 模块 的 概述 
国 ) 考题 信息 管理 模块 的 技术 分 析 
国 ; 考题 信息 添加 的 实现 过 程 
国 查询 考题 信息 的 实现 过 程 





国 开 发 技巧 、 难 点 分 析 与 Ajax 无 刷新 技术 


国 考生 登录 编号 的 获取 


国 ) 通过 Ajax 技术 实现 计时 与 显示 剩余 时 间 





仁 ] Ajax 的 工作 原理 
仁 ] Ajax 的 工作 流程 
三] Ajax 中 的 核心 技术 XMLHttpRequest 


项 目 6 物流 配送 信息 网 


日 


国 开发 背景 和 系统 分 析 
但 ] 开发 背景 
但 ] 需求 分 析 
仁 ] 可 行 性 分 析 


口 国 系 统 设计 


国 系统 目标 

下 系统 功能 结构 

便 系统 预览 

恒 开发 环境 

国 文件 夹 组 织 结构 
国 数 据 库 设计 
国 数据 库 分 析 
国 数据 库 概念 设计 
国 ; 创建 数据 库 及 数据 表 
国 网 站 首页 设计 
国 车 源 信息 查询 模块 设计 
国 发 货 单 管理 模块 设计 











习 国 回执 单 验 收 管理 模块 设计 
日 国 基 础 信息 管理 模块 设计 
基础 信息 管理 模块 概述 
基础 信息 管理 模块 技术 分 析 
客户 信息 管理 的 实现 过 程 
车 源 信息 管理 的 实现 过 程 
5 国 开 发 技巧 与 难点 分 析 
应 用 存储 过 程 实现 管理 员 登 录 
应 用 正则 表达 式 验证 电话 号 码 
5 国 报表 打印 技术 


项 目 7 学校 图 书馆 管理 系统 
日 国 开 发 背景 和 需求 分 析 
开发 背景 
需求 分 析 
5 国 系统 设计 
系统 目标 
系统 功能 结构 
系统 流程 图 
系统 预览 
开发 环境 
文件 夹 组 织 结构 
5 国 数据 库 设计 
数据 库 分 析 
数据 库 概 念 设计 
创建 数据 库 及 数据 表 
日 国 首 页 设计 
5 国 管 理 员 模块 设计 
管理 员 模 块 概述 
管理 员 模 块 技术 分 析 
系统 登录 的 实现 过 程 
查看 管理 员 的 实现 过 程 
添加 管理 员 的 实现 过 程 
设置 管理 员 权限 的 实现 过 程 
删除 管理 员 的 实现 过 程 
单元 测试 
5 国 图 书 档案 管理 模块 设计 
图 书 档案 管理 模块 概述 
图 书 档案 管理 模块 技术 分 析 
查看 图 书信 息 列 表 的 实现 过 程 
添加 图 书信 息 的 实现 过 程 
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国 修改 图 书信 息 的 实现 过 程 
国 删除 图 书信 息 的 实现 过 程 
书 借 还 模块 设计 






B 续 借 的 实现 过 程 

节 归 还 的 实现 过 程 

书 借阅 查询 的 实现 过 程 
国 单元 测试 

口 国 开 发 技巧 、 难 点 分 析 与 联接 语句 技术 
国 如 何 自动 计算 图 书 归还 日 期 
国 | 如 何 对 图 书 借阅 信息 进行 统计 排行 
国 联接 语句 技术 


项 目 8 博客 管理 系统 
日 国 开 发 背景 和 需求 分 析 
国 开发 背景 
国 需求 分 析 
日 国 系统 设计 
镜 系统 目标 
国 ] 系统 功能 结构 
系统 功能 预览 
系统 流程 图 
9 开发 环境 
直 文件 夹 组 织 结构 
口 国 数 据 库 设计 
国 数据 库 分 析 
国 数据 库 概念 设计 
国 数据 库 物理 结构 设计 
口 国 首 页 设计 
口 国文 章 管理 模块 设计 
国 | 文章 管理 模块 概述 
有 | 文章 管理 模块 技术 分 析 
国 ) 添加 文章 的 实现 过 程 
国 文章 列表 的 实现 过 程 
国 查看 文章 、 评 论 的 实现 过 程 
国 删除 文章 、 评 论 的 实现 过 程 
日 国 图 片上 传 模块 设计 











天 | 图 片上 传 模块 概述 








片上 传 的 实现 过 程 
图 片 浏览 的 实现 过 程 
删除 图 片 的 实现 过 程 
单元 测试 

口 国有 朋友 图 模块 设计 

局 国 开发 技巧 与 难点 分 析 

5 国 登录 验证 码 技术 专题 
简单 的 数字 验证 
数字 图 形 验 证 码 
汉字 图 形 验 证 码 


项 目 9 企业 快 信 系统 
口 国 开 发 背景 和 系统 分 析 
开发 背景 
需求 分 析 
可 行 性 分 析 
5 国 系 统 设计 
系统 目标 
系统 功能 结构 
系统 流程 图 
系统 预览 
开发 环境 
文件 夹 组 织 结构 
口 国 SMTP 和 POP3 服务 器 的 安装 与 配置 
SMTP 服务 器 的 安装 和 配置 
POP3 服务 器 的 安装 和 配置 
5 国 数据 库 设 计 
数据 库 概念 设计 
创建 数据 库 及 表 
数据 库 逻 辑 结构 设计 
5 国 首页 设计 
口 国 系统 信息 管理 模块 设计 
局 国 发 送 短信 模块 设计 
5 国 连接 邮件 接口 模块 的 设计 
日 国 接 收 邮件 模块 设计 
接收 邮件 模块 概述 
接收 邮件 模块 技术 分 析 
查看 邮件 的 实现 过 程 
删除 邮件 的 实现 过 程 
下 载 附件 的 实现 过 程 
口 国 发 送 邮件 模块 设计 


光盘 “开发 资源 库 ” 目 录 





发 送 邮件 模块 概述 PHP 与 Access 数据 库 的 连接 
发 送 邮件 模块 技术 分 析 邮件 群发 技术 
发 送 邮件 的 实现 过 程 习 国 通 过 fsockopen0 函 数 发 送 短信 技术 
查看 邮件 记录 的 实现 过 程 项 目 10 online 影视 365 网 
口 国 开 发 技巧 与 难点 分 析 ee 





第 4 大 部 分 “能 力 测试 题库 
(626 道 能 力 测试 题目 ， 光 盘 路 径 : 开发 资源 库 /能 力 测试 


第 1 部 分 PHP 编程 基础 能 力 测试 高 级 测试 


口 英语 能 力 测 |i 
第 2 部 分 数学 及 逻辑 思维 能 力 测试 第 3 部 分 编程 英语 能 力 测试 
国 基本 测试 英语 基础 能 力 测试 


国 进 阶 测试 国 英语 进 阶 能 力 测试 


第 5 大 部 分 “面试 资源 库 


(342 项 面试 真题 ， 光 盘 路 径 : 开发 资源 库 / 面 试 系统 ) 


第 1 部 分 PHP 程序 员 职 业 规划 面向 对 象 面试 真题 
国 你 了 解 程序 员 吗 错误 与 异常 处 理 面试 真题 
国 程序 员 自 我 定位 SQL 语言 面试 真题 
pe MySQL 数据 库 面试 真题 
第 2 部 分 PHP 程序 员 面 试 技巧 服务 器 、 操 作 系 统 与 网 络 面试 真题 
国 面试 的 三 种 方式 项 目 设计 面试 真题 
国 如 何 应 对 企业 面试 
国 英语 面试 第 4 部 分 PHP 企业 面试 真题 汇编 
国 电话 面试 企业 面试 真题 汇编 (一 ) 
wm 时 
第 3 部 分 PHP 常见 面试 题 企业 面试 真题 汇编 (四) 
王 Web 页 面 设计 面试 真题 第 5 部 分 PHP 虚拟 面试 系统 
国 PHP 语言 基础 面试 真题 ea 








畜 
名; 











Wm 第 1 章 数据 库 基础 

MW 第 2 章 初 识 MySQL 

H 第 3 章 使 用 MySQL 图 形 化 管理 工具 
Wp 第 4 章 数据 库 操作 

Wm 第 5 章 存储 引擎 及 数据 类 型 

MW 第 6 章 操作 数据 表 


本 篇 通过 对 数据 库 基础 、 初 识 My5QL. 使 用 My5QL 图 形 化 管理 工具 、 数 据 库 
操作 、 存 储 引 擎 及 数据 类 型 和 操作 数据 表 等 内 容 的 介绍 ， 并 结合 大 量 的 图 示 、 举 例 
和 视频 等 帮助 读者 快速 掌握 MySQL， 并 为 以 后 的 知识 黄 定 坚实 的 基础 。 


第 章 


数据 库 基 础 


( 盈 。 视频 讲解 .25 分 钟 ) 


本 章 主要 介绍 数据 库 的 相关 概念， 主要 包括 数据 库 系 统 概述 、 数 据 模型 和 数据 
库 的 体系 结构 。 通 过 本 章 的 学 习 ， 读 者 应 该 掌 提 数据库 系 统 、 数 据 模型 、 数 据 库 三 
级 模式 结构 以 及 数据 库 规范 化 等 概念 。 
通过 阅读 本 章 ， 读 者 可 以 : 
了 解数 据 库 技术 的 发 展 史 
掌握 数据 库 系统 的 组 成 
熟悉 数据 模型 
掌握 关系 数据 库 
掌握 数据 库 的 体系 结构 


豆 吾 吾 吾 至 
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1.1 数据 库 系统 概述 


1.1.1 数据 库 技术 的 发 展 


数据 库 技术 是 应 数据 管理 任务 的 需求 而 产生 的 ， 随 着 计算 机 技术 的 发 展 ， 对 数据 管理 技术 也 
不 断 地 提出 更 高 的 要 求 ， 其 先后 经 历 了 人 工 管理 、 文 件 系统 和 数据 库 系统 3 个 阶段 ， 下 面 分 别 进 
行 介绍 。 

1， 人工 管理 阶段 


20 世纪 50 年 代 中 期 以 前 , 计算 机 主要 用 于 科学 计算 。 当 时 硬件 和 软件 设备 都 很 落后 ,数据 基本 依 
赖 于 人 工 管理 。 人 工 管理 阶段 具有 如 下 特点 。 

(1) 数据 不 保存 。 

(2) 使 用 应 用 程序 管理 数据 。 

(3) 数据 不 共享 。 

(4) 数据 不 具有 独立 性 。 


2. 文件 系统 阶段 


20 世纪 50 年 代 后 期 到 20 世纪 60 年 代 中 期 , 硬件 和 软件 技术 都 有 了 进一步 发 展 , 有 了 磁盘 等 存储 
设备 和 专门 的 数据 管理 软件 〈 即 文件 系统 ) 。 该 阶段 具有 如 下 特点 。 

(1) 数据 可 以 长 期 保存 。 

(2) 由 文件 系统 管理 数据 。 

(3) 共享 性 差 ， 数 据 元 余 大 。 

(4) 数据 独立 性 差 。 


3. 数据 库 系 统 阶段 
20 世纪 60 年 代 后 期 以 来 ， 计 算 机 应 用 于 管理 系统 ， 而 且 规模 越 来 越 大 ， 应 用 越 来 越 广泛 ， 数 据 量 
急剧 增长 ， 对 共享 功能 的 要 求 越 来 越 强 烈 ， 这 样 使 用 文件 系统 管理 数据 已 经 不 能 满足 要 求 ， 于 是 出 现 


了 数据 库 系 统 来 统一 管理 数据 。 数 据 库 系统 的 出 现 ， 满 足 了 多 用 户 、 多 应 用 共享 数据 的 需求 ， 比 文件 
系统 具有 明显 的 优点 ， 标 志 着 数据 管理 技术 的 飞跃 。 


1.1.2 ”数据 库 系 统 的 组 成 
数据 库 系 统 (DataBase System，DBS) 是 采用 数据 库 技术 的 计算 机 系统 ， 是 由 数据 库 (数据 ) 、 


数据 库 管理 系统 、 数 据 库 管理 员 (人员) 、 支 持 数 据 库 系统 的 硬件 和 软件 (应 用 开发 工具 、 应 用 系 
统 等 ) 以 及 用 户 5 部 分 构成 的 运行 实体 ,如 图 1.1 所 示 。 其 中 ,数据库 管理 员 (DataBase Administrator， 
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DBA) 是 对 数据 库 进行 规划 、 设 计 、 维 护 和 监视 等 的 专业 管理 人 员 ， 在 数据 库 系 统 中 起 着 非常 重要 
的 作用 。 






































用 户 ] [ 用户] [用户 ] 
应 用 系统 
应 用 开发 工具 > 
《数据 库 管 理 系统 


图 1.1 数据 库 系统 的 组 成 


1.2 数据 模型 


1.2.1 数据 模型 的 概念 


数据 模型 是 数据 库 系统 的 核心 与 基础 ， 是 关于 描述 数据 与 数据 之 间 的 联系 、 数 据 的 语义 、 数 据 一 
致 性 约束 的 概念 性 工具 的 集合 。 

数据 模型 通常 是 由 数据 结构 、 数 据 操 作 和 完整 性 约束 3 部 分 组 成 的 ， 分 别 如 下 。 

(1) 数据 结构 :是 对 系统 静态 特征 的 描述 ， 描 述 对 象 包括 数据 的 类 型 、 内 容 、 性 质 和 数据 之 间 的 
相互 关系 。 

(2) 数据 操作 : 是 对 系统 动态 特征 的 描述 ， 是 对 数据 库 各 种 对 象 实例 的 操作 。 

(3) 完整 性 约束 : 是 完整 性 规则 的 集合 ， 它 定义 了 给 定数 据 模型 中 数据 及 其 联系 所 具有 的 制约 和 
依存 规则 。 


1.2.2 ”常见 的 数据 模型 


常用 的 数据 库 数据 模型 主要 有 层次 模型 、 网 状 模型 和 关系 模型 ， 下 面 分 别 进行 介绍 。 

(1) 层次 模型 : 用 树 状 结构 表示 实体 类 型 及 实体 间 联 系 的 数据 模型 称 为 层次 模型 ， 如 图 1.2 所 示 。 
它 具 有 以 下 特点 。 

Q@ 每 棵 树 有 且 仅 有 一 个 无 双亲 节点 ， 称 为 根 。 

@ 树 中 除根 外 所 有 节点 有 且 仅 有 一 个 双亲 。 
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用 Pal 思 库 [用 Pui | [WPu2 | [RFU 













外 模式 











概 全 级 数据 库 


一 DBA 视图 





图 1.2 层次 模型 


(2) 网 状 模型 : 用 有 向 图 结构 表示 实体 类 型 及 实体 间 联 系 的 数据 模型 称 为 网 状 模型 ， 如 图 1.3 所 
示 。 用 网 状 模型 编写 应 用 程序 极其 复杂 ， 数 据 的 独立 性 较 差 。 





















































A B A A 
a en se 
| | /| B B 
| I I 
C D C C 
图 1.3 网 状 模型 


(3) 关系 模型 : 以 二 维 表 来 描述 数据 。 关 系 模 型 中 ， 每 个 表 有 多 个 字段 列 和 记录 行 ， 每 个 字段 列 
有 固定 的 属性 (数字 、 字 符 、 日 期 等 ) ， 如 图 1.4 所 示 。 关 系 模型 数据 结构 简单 、 清 晰 、 具 有 很 高 的 数 
据 独 立 性 ， 是 目前 主流 的 数据 库 数据 模型 。 

关系 模型 的 基本 术语 如 下 。 

Q 关系 : 一 个 二 维 表 就 是 一 个 关系 。 

@ 元 组 : 二 维 表 中 的 一 行 ， 即 表 中 的 记录 。 

图 属性 : 二 维 表 中 的 一 列 ， 用 类 型 和 值 表示 。 

@ 域 : 每 个 属性 取 值 的 变化 范围 ， 如 性 别 的 域 为 { 男 ， 女 }。 

关系 中 的 数据 约束 如 下 。 

Q 实体 完整 性 约束 : 约束 关系 的 主键 中 属性 值 不 能 为 空 值 。 

@ 参照 完整 性 约束 : 关系 之 间 的 基本 约束 。 
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图 用 户 定义 的 完整 性 约束 : 反映 了 具体 应 用 中 数据 的 语义 要 求 。 
























































学 生 信息 表 
学 生 姓名 年 级 家 庭 住址 
张 三 2000 成 都 
李 四 2000 北京 
王 五 2000 上 海 
成 绩 表 
学 生 姓 名 课程 成 绩 
张 三 数学 100 
张 三 物理 95 
张 三 社会 90 
李 四 数学 85 
李 四 社会 90 
玉 五 数学 80 
王 五 物理 75 
图 1.4 关系 模型 


1.2.3 ”关系 数据 库 的 规范 化 


关系 数据 库 的 规范 化 理论 为 : 关系 数据 库 中 的 每 一 个 关系 都 要 满足 一 定 的 规范 。 根 据 满足 规范 的 
条 件 不 同 ， 可 以 分 为 5 个 等 级 : 第 一 范式 (INF) 、 第 二 范式 (2NF) …… 第 五 范式 CSNF) 。 其 中 ， 
NF 是 Normal Form 的 缩写 。 一 般 情况 下 ， 只 要 把 数据 规范 到 第 三 范式 标准 就 可 以 满足 需要 了 。 下 面 举 
例 介绍 前 3 种 范式 。 


1. 第 一 范式 (1NF) 


在 一 个 关系 中 ， 消 除 重复 字段 ， 且 各 字段 都 是 最 小 的 逻辑 存储 单位 。 第 一 范式 是 第 二 和 第 三 范式 
的 基础 ， 是 最 基本 的 范式 。 第 一 范式 包括 下 列 指导 原则 。 

(1) 数据 组 的 每 个 属性 只 可 以 包含 一 个 值 。 

(2) 关系 中 的 每 个 数组 必须 包含 相同 数量 的 值 。 

(3) 关系 中 的 每 个 数组 一 定 不 能 相同 。 

在 任何 一 个 关系 数据 库 中 ， 第 一 范式 是 对 关系 模式 的 基本 要 求 ， 不 满足 第 一 范式 的 数据 库 就 不 是 
关系 型 数据 库 。 

如 果 数 据 表 中 的 每 一 个 列 都 是 不 可 再 分 割 的 基本 数据 项 一 一 即 同一 列 中 不 能 有 多 个 值 ， 那 么 就 称 
此 数据 表 符合 第 一 范式 ， 由 此 可 见 第 一 范式 具有 不 可 再 分 解 的 原子 特性 。 

在 第 一 范式 中 ， 数 据 表 的 每 一 个 行 只 包含 一 个 实体 的 信息 ， 并 且 每 一 行 的 每 一 列 只 能 存放 实体 的 
一 个 属性 。 例 如 ， 对 于 学 生 信息 ， 不 可 以 将 学 生 实体 的 所 有 属性 信息 (如 学 号 、 姓 名 、 性 别 、 年 龄 、 
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班级 等 ) 都 放 在 一 个 列 中 显示 ， 也 不 能 将 学 生 实体 的 两 个 或 多 个 属性 信息 放 在 一 个 列 中 显示 ， 学 生 实 
体 的 每 个 属性 信息 都 放 在 一 个 列 中 显示 。 

如 果 数据 表 中 的 列 信息 都 符合 第 一 范式 ， 那 么 在 数据 表 中 的 字段 都 是 单一 的 、 不 可 再 分 的 。 如 
表 1.1 就 是 不 符合 第 一 范式 的 学 生 信息 表 ， 因 为 “班级 ” 列 中 包含 “ 系 别 ” 和 “班级 ”两 个 属性 信息 ， 
这 样 “ 班 级 ” 列 中 的 信息 就 不 是 单一 的 ， 是 可 以 再 分 的 ; 而 表 1.2 就 是 符合 第 一 范式 的 学 生 信息 表 ， 它 
将 原 “ 班 级 ” 列 的 信息 拆 分 到 “ 系 别 ” 列 和 “班级 ” 列 中 。 


表 1.1 不 符合 第 一 范式 的 学 生 信息 表 


学 号 姓 名 性 别 年 龄 班级 
9527 东 * 方 男 20 计算 机 系 3 班 


表 1.2 符合 第 一 范式 的 学 生 信息 表 


学 号 | 姓名 | 性 别 | 年龄 | 系 别 | 班 级 

9527 东 * 方 | 男 | 2 | 计算 机 | 3 班 

2. 第 二 范式 (2NF) 

第 二 范式 是 在 第 一 范式 的 基础 上 建立 起 来 的 ， 即 满足 第 二 范式 必 先 满足 第 一 范式 (INF) 。 第 二 
范式 要 求 数 据 库 表 中 的 每 个 实体 〈 即 各 个 记录 行 ) 必须 可 以 被 唯一 地 区 分 。 为 实现 区 分 各 行 记录 通常 
需要 为 表 设 置 一 个 “区 分 列 ”， 用 以 存储 各 个 实体 的 唯一 标识 。 在 学 生 信息 表 中 ， 设 置 了 “学 号 ” 列 ， 
由 于 每 个 学 生 的 编号 都 是 唯一 的 ， 因 此 每 个 学 生 可 以 被 唯一 地 区 分 〈 即 使 学 生存 在 重 名 的 情况 下 ) ， 
那么 这 个 唯一 属性 列 被 称 为 主 关 键 字 或 主键 。 

第 二 范式 要 求实 体 的 属性 完全 依赖 于 主 关键 字 ， 即 不 能 存在 仅 依赖 主 关键 字 一 部 分 的 属性 ， 如 果 
存在 ， 那 么 这 个 属性 和 主 关键 字 的 这 一 部 分 应 该 分 离 出 来 形成 一 个 新 的 实体 ， 新 实体 与 原 实体 之 问 是 
一 对 多 的 关系 。 

例如 ， 这 里 以 “员工 工资 信息 表 ” 为 例 ， 若 以 〈 员 工 编码 、 岗 位 ) 为 组 合 关键 字 〈 即 复合 主键 ) ， 
就 会 存在 如 下 决定 关系 。 

(员工 编码 ， 岗 位 ) 一 〈 决 定 ) 〈 姓 名 、 年 龄 、 学 历 、 基 本 工资 、 绩 效 工 资 、 奖 金 ) 

在 上 面 的 决定 关系 中 ， 还 可 以 进一步 拆 分 为 如 下 两 种 决定 关系 。 

(员工 编码 ) 小 决定 ) 〈 姓 名 、 年 龄 、 学 历 ) 

(岗位 ) 一“〈 决 定 ) 〈 基 本 工资 ) 

其 中 ， 员 工 编码 决定 了 员工 的 基本 信息 (包括 姓名 、 年 龄 、 学 历 等 ) ; 而 岗位 决定 了 基本 工资 ， 
所 以 这 个 关系 表 不 满足 第 二 范式 。 

对 于 上 面 的 这 种 关系 ， 可 以 把 上 述 两 个 关系 表 更 改 为 如 下 3 个 表 。 

员工 档案 表 : EMPLOYEE (员工 编码 、 姓 名 、 年 龄 和 学 历 ) 

岗位 工资 表 : QUARTERS (岗位 和 基本 工资 》 

员工 工资 表 : PAY (员工 编码 、 岗 位 、 绩 效 工 资 和 奖金 ) 
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3. 第 三 范式 (3NF) 


第 三 范式 是 在 第 二 范式 的 基础 上 建立 起 来 的 ， 即 满足 第 三 范式 必 先 满足 第 二 范式 。 第 三 范式 要 求 
关系 表 不 存在 非 关键 字 列 对 任意 候选 关键 字 列 的 传递 函数 依赖 ， 也 就 是 说 ， 第 三 范式 要 求 一 个 关系 表 
中 不 包含 已 在 其 他 表 中 包含 的 非 主 关键 字 信 息 。 

所 谓 传递 函数 依赖 ， 就 是 指 如 果 存在 关键 字段 A 决定 非 关键 字段 B， 而 非 关键 字段 B 决定 非 关键 
字段 C， 则 称 非 关键 字段 C 传递 函数 依赖 于 关键 字段 A。 

例如 ， 这 里 以 员工 信息 表 (EMPLOYEE) 为 例 ， 该 表 中 包含 员工 编号 、 员 工 姓 名 、 年 龄 、 部 门 纺 
码 、 部 门 经 理 等 信息 ， 该 关系 表 的 关键 字 为 “员工 编号 ”， 因 此 存在 如 下 决定 关系 。 

《员工 编码 ) -y (决定) 《员工 姓名 、 年 龄 、 部 门 编码 、 部 门 经 理 ) 

上 面 的 这 个 关系 表 是 符合 第 二 范式 的 ， 但 它 不 符合 第 三 范式 ， 因 为 该 关系 表 内 部 隐 含 着 如 下 决定 
关系 。 





(员工 编码 ) 一 〈 决 定 ) 〈 部 门 编码 ) 了 决定 ) 部门 经 理 ) 

上 面 的 关系 表 存 在 非 关键 字段 “部 门 经 理 ” 对 关键 字段 “员工 编码 ”的 传递 函数 依赖 。 对 于 上 面 
的 这 种 关系 ， 可 以 把 这 个 关系 表 (EMPLOYEE) 更改 为 如 下 两 个 关系 表 。 

员工 信息 表 : EMPLOYEE (员工 编码 、 员 工 姓名 、 年 龄 和 部 门 编码 》 

部 门 信息 表 : DEPARTMENT (部 门 编码 和 部 门 经 理 ) 

对 于 关系 型 数据 库 的 设计 ， 理 想 的 设计 目标 是 按照 “规范 化 ”原则 存储 数据 ， 因 为 这 样 做 能 够 消 
除数 据 元 余 、 更 新 异常 、 插 入 异常 和 删除 异常 。 





1.2.4 ”关系 数据 库 的 设计 原则 


数据 库 设计 是 指 对 于 一 个 给 定 的 应 用 环境 ， 根 据 用 户 的 需求 ， 利 用 数据 模型 和 应 用 程序 模拟 现实 
世界 中 该 应 用 环境 的 数据 结构 和 处 理 活动 的 过 程 。 

数据 库 设 计 原则 如 下 。 

(1) 数据 库 内 数据 文件 的 数据 组 织 应 获得 最 大 限度 的 共享 、 最 小 的 元 余 度 ， 消 除数 据 及 数据 依赖 
关系 中 的 元 余部 分 ， 使 依赖 于 同一 个 数据 模型 的 数据 达到 有 效 的 分 离 。 

(2) 保证 输入 、 修 改 数 据 时 数据 的 一 致 性 与 正确 性 。 

(3) 保证 数据 与 使 用 数据 的 应 用 程序 之 间 的 高 度 独立 性 。 


1.2.5 ”实体 与 关系 


实体 是 指 客观 存在 并 可 相互 区 别 的 事物 ， 实 体 既 可 以 是 实际 的 事物 ， 也 可 以 是 抽象 的 概念 或 关系 。 

实体 之 间 有 3 种 关系 ， 分 别 如 下 。 

(1) 一 对 一 关系 : 是 指 表 A 中 的 一 条 记录 在 表 B 中 有 且 只 有 一 条 相 匹 配 的 记录 。 在 一 对 一 关系 
中 ， 大 部 分 相关 信息 都 在 一 个 表 中 。 
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(2) 一 对 多 关系 : 是 指 表 A 中 的 行 可 以 在 表 B 中 有 许多 匹配 行 ， 但 是 表 B 中 的 行 只 能 在 表 A 中 
有 一 个 匹配 行 。 
(3) 多 对 多 关系 : 是 指 关 系 中 每 个 表 的 行 在 相关 表 中 具有 多 个 匹配 行 。 在 数据 库 中 ， 多 对 多 关系 


的 建立 是 依靠 第 三 个 表 〈 称 作 连 接 表 ) 实现 的 ， 连 接 表 包 含 相关 的 两 个 表 的 主键 列 ， 然 后 从 两 个 相关 
表 的 主键 列 分 别 创建 与 连接 表 中 的 匹配 列 的 关系 。 


1.3 数据 库 的 体系 结构 


1.3.1 数据 库 三 级 模式 结构 


数据 库 的 三 级 模式 结构 是 指 模式 、 外 模式 和 内 模式 。 

1. 模式 

模式 也 称 罗 辑 模式 或 概念 模式 ， 是 数据 库 中 全 体 数据 的 逻辑 结构 和 特征 的 描述 ， 是 所 有 用 户 的 公 
共 数 据 视图 。 一 个 数据 库 只 有 一 个 模式 。 模 式 处 于 三 级 结构 的 中 间 层 。 


$6 注意 
定义 模式 时 不 仅 要 定义 数据 的 逻辑 结构 ,而且 要 定义 数据 之 间 的 联系 ,定义 与 数据 有 关 的 安全 
性 、 完 整 性 要 求 。 


2. 外 模式 


外 模式 也 称 用 户 模式 ， 是 数据 库 用 户 〈 包 括 应 用 程序 员 和 最 终 用 户 ) 能 够 看 见 和 使 用 的 局 部 数据 
的 逻辑 结构 和 特征 的 描述 ， 是 数据 库 用 户 的 数据 视图 ， 是 与 某 一 应 用 有 关 的 数据 的 逻辑 表示 。 外 模式 
是 模式 的 子 集 ， 一 个 数据 库 可 以 有 多 个 外 模式 。 


0 税 明 


外 模式 是 保证 数据 安全 性 的 一 个 有 力 措施 。 


3. 内 模式 


内 模式 也 称 存储 模式 ， 是 数据 物理 结构 和 存储 方式 的 描述 ， 是 数据 在 数据 库 内 部 的 表示 方式 。 一 
个 数据 库 只 有 一 个 内 模式 。 


1.3.2 三 级 模式 之 间 的 映射 


为 了 能 够 在 内 部 实现 数据 库 的 3 个 抽象 层次 的 联系 和 转换 ， 数 据 库 管理 系统 在 三 级 模式 之 间 提 供 
了 两 层 映射 ， 分 别 为 外 模式 /模式 映射 和 模式 /内 模式 映射 ， 下 面 分 别 介 绍 。 
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1. 外 模式 /模式 映射 


对 于 同一 个 模式 可 以 有 任意 多 个 外 模式 。 对 于 每 一 个 外 模式 ， 数 据 库 系 统 都 有 一 个 外 模式 /模式 映 
射 。 当 模式 改变 时 ， 由 数据 库 管理 员 对 各 个 外 模式 /模式 映射 做 相应 的 改变 ， 可 以 使 外 模式 保持 不 变 。 
这 样 ， 依 据 数 据 外 模式 编写 的 应 用 程序 就 不 用 修改 ， 保 证 了 数据 与 程序 的 逻辑 独立 性 。 

2. 模式 /内 模式 映射 

数据 库 中 只 有 一 个 模式 和 一 个 内 模式 ， 所 以 模式 /内 模式 映射 是 唯一 的 ， 它 定义 了 数据 库 的 全 局 逻 
辑 结构 与 存储 结构 之 间 的 对 应 关系 。 当 数据 库 的 存储 结构 改变 时 ， 由 数据 库 管理 员 对 模式 /内 模式 映射 
做 相应 改变 ， 可 以 使 模式 保持 不 变 ， 应 用 程序 相应 地 也 不 做 变动 。 这 样 ， 保 证 了 数据 与 程序 的 物理 独 
立 性 。 


1.4 小 结 


本 章 主 要 介绍 的 是 数据 库 技术 中 的 一 些 基本 概念 和 原理 ， 其 中 重点 包括 数据 库 技术 的 发 展 、 数 据 
库 系统 的 组 成 、 数 据 模型 的 概念 、 常 见 的 数据 模型 、 关 系数 据 库 的 规范 化 及 设计 原则 、 实 体 与 关系 、 
数据 库 的 三 级 模式 结构 ， 以 及 三 级 模式 之 间 的 映射 等 内 容 。 其 中 ， 常 见 的 数据 模型 和 关系 数据 库 的 规 
范 化 及 设计 原则 希望 读者 认真 学 习 ， 重 点 掌握 。 


1.5 实践 与 练习 


1. 数据 库 技术 的 发 展 经 历 了 哪 3 个 阶段 ? 
2. 数据 模型 通常 是 由 哪 3 部 分 组 成 的 ? 
3. 常用 的 数据 库 数据 模型 主要 有 哪 几 种 ? 
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第 章 


初 识 MySQL 
(全 视频 讲解 : 12 分 钟 ) 


My5QL 数据 库 可 以 称 得 上 是 目前 运行 速度 最 快 的 SQL 数据库。 除了 有 具有 许多 
其 他 数据 库 所 不 具备 的 功能 和 选择 之 外 ，MySQL 数据 库 还 是 一 种 完全 免费 的 产品 ， 
用 户 可 以 直接 从 网 上 下 载 使 用 ， 不 必 支 付 任何 费用 。 另 外 ，MySQL 数据 库 的 路 平 
台 性 也 是 一 大 优势 。 本 章 将 对 My5QL 教 据 库 的 概念 、MySQL 的 将 性 、 应 用 环境 ， 
以 及 如 何 安装 、 配 置 、 启 动 、 连 接 、 断 开 和 停止 My5QL 服务 器 进行 详细 介绍 。 
通过 阅读 本 章 ， 读 者 可 以 : 
了 解 MySQL 数据 库 的 概念 及 其 优势 
熟悉 MySQL 的 特性 
了 解 MySQL 的 应 用 环境 
掌握 如 何 安装 和 配置 MySQL 服务 路 
掌握 启动 、 连 接 、 断 开 和 停止 MySQL 服务 路 的 方法 
了 解 如 何 学 好 MySQL 


于 于 于 至 至 至 


MySQL 从 入 门 到 精通 


2.1 了 解 MySQL 


MySQL 是 目前 最 为 流行 的 开放 源 代 码 的 数据 库 管 理 系统 ， 是 完全 网 络 化 的 、 跨 平台 的 关系 型 数据 
库 系统 ， 它 是 由 瑞典 的 MySQL AB 公司 开发 的 ， 由 MySQL 的 初始 开发 人 员 David Axmark 和 Michael 
"Monty" Widenius 于 1995 年 建立 ， 目 前 属于 Oracle 公司 。 它 的 象征 符号 是 一 只 名 为 Sakila 的 海豚 ， 代 
表 着 MySQL 数据 库 和 团队 的 速度 、 能 力 、 精 确 和 优秀 本 质 。 

MySQL 数据 库 可 以 称 得 上 是 目前 运行 速度 最 快 的 SQL 数据 库 。 除 了 具有 许多 其 他 数据 库 所 不 具 
备 的 功能 和 选择 之 外 ，MySQL 数据 库 还 是 一 种 完全 免费 的 产品 ， 用 户 可 以 直接 从 网 上 下 载 使 用 ， 不必 
支付 任何 费用 。 


2.1.1 MySQL 数据 库 的 概念 


数据 库 (Database) 就 是 一 个 存储 数据 的 仓库 。 为 了 方便 数据 的 存储 和 管理 ， 它 将 数据 按照 特定 的 
规律 存储 在 磁盘 上 。 通 过 数据 库 管理 系统 ， 可 以 有 效 地 组 织 和 管理 存储 在 数据 库 中 的 数据 。MySQL 就 
是 这 样 的 一 个 关系 型 数据 库 管理 系统 (RDBMS) ， 它 可 以 称 得 上 是 目前 运行 速度 最 快 的 SQL 数据 库 
管理 系统 。 


2.1.2 ”MySQL 的 优势 


MySQL 是 一 款 自由 软件 ， 任 何人 都 可 以 从 其 官方 网 站 下 载 。MySQL 是 一 个 真正 的 多 用 户 、 多 线 
程 SQL 数据 库 服务 器 。 它 是 以 客户 /服务 器 结构 的 实现 ， 由 一 个 服务 器 守护 程序 mysqld 和 很 多 不 同 的 
客户 程序 和 库 的 组 成 。 它 能 够 快捷 ` 有 效 和 安全 地 处 理 大 量 的 数据 。 相 对 于 Oracle 等 数据 库 来 说 , MySQL 
在 使 用 时 非常 简单 。MySQL 的 主要 目标 是 快捷 、 便 捷 和 易 用 。 

MySQL 被 广泛 地 应 用 在 Internet 上 的 中 小 型 网 站 中 。 由 于 其 体积 小 、 速 度 快 、 总 体 拥 有 成 本 低 ， 
尤其 是 开放 源 代 码 这 一 特点 ， 成 为 多 数 中 小 型 网 站 为 了 降低 网 站 总 体 拥有 成 本 而 选择 MySQL 作为 网 
站 数据 库 的 重要 指标 。 





2.1.3 ”MySQL 的 发 展 史 


MySQL 这 个 名 字 的 由 来 已 经 无 从 考究 了 。 基 本 指南 以 及 大 量 的 库 和 工具 采用 前 红 My， 已 经 有 10 
年 以 上 了 ; 另外 ，MySQL 的 创始 人 之 一 的 Monty Widenius 的 女儿 也 叫 My。 到 底 哪个 是 MySQL 名 字 
的 由 来 ， 至 今 仍 是 一 个 谜 ， 包 括 开 发 者 也 不 知道 。 

MySQL 的 海豚 徽标 的 名 字 为 Sakila， 它 是 由 MySQL AB 公司 的 创办 人 从 用 户 在 “Dolphin 命名 ” 
比赛 中 提供 的 众多 建议 中 选 定 的 ， 是 由 来 自 非洲 斯 威 士 兰 的 开放 源码 软件 开发 人 Ambrose Twebaze 提 
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出 的 。 根 据 Ambrose 的 说 法 ， 按 斯 威 士 兰 的 本 地 语言 ， 女 性 化 名 称 Sakila 源 自 SiSwati。Sakila 也 是 坦 
桑 尼 亚 、Arusha 地 区 的 一 个 镇 的 镇 名 ， 靠 近 Ambrose 的 母国 乌干达 。 

MySQL 从 无 到 有 ， 到 技术 的 不 断 更 新 、 版 本 的 不 断 升 级 ， 经 历 了 一 个 漫长 的 过 程 ， 这 个 过 程 是 实 
践 的 过 程 ， 是 MySQL 成 长 的 过 程 。 时 至 今日 ，MySQL 的 版 本 已 经 更 新 到 了 MySQL 5.6。 如 图 2.1 所 
示 为 MySQL 官方 网 站 上 的 截图 ， 足 以 反映 出 MySQL 的 成 长 历程 。 


MySQL Server 


MySQL 5.6 Reference Manual 
MySQL 5.5 Reference Manual 
MySQL 5.1 Reference Manual 
MySQL 5.0 Reference Manual 


MySQL 3.23/4.0/4.1 Manual 





2.1 MySQL 版 本 的 发 展 


/ 
2 曙 
2008 年 1 月 16 日 ,MySQL 被 Sun 公司 收购 ， 而 到 2009 年 ，Sun 公司 被 世界 第 二 大 软件 供应 
商 Oracle 公司 收购 ， 成 为 Oracle 数据 库 的 有 益 补 充 。 





2.2 MySQL 的 特性 


MySQL 是 一 个 真正 的 多 用 户 、 多 线程 SQL 数据 库 服 务 器 。SQL (结构 化 查询 语言 ) 是 世界 上 最 

流行 的 和 标准 化 的 数据 库 语言 。 下 面 看 一 下 MySQL 的 特性 。 

(1) 使 用 C 和 C++ 语言 编写 ， 并 使 用 了 多 种 编译 器 进行 测试 ， 保 证 源 代 码 的 可 移植 性 。 

(2) 支持 AIX、FreeBSD、HP-UX、Linux、Mac OS、Novell Netware、OpenBSD、OS/2 Wrap、 
Solaris、Windows 等 多 种 操作 系统 。 

(3) 为 多 种 编程 语言 提供 了 API。 这 些 编程 语言 包括 C、C++、Python、Java、Perl、PHP、Eiffel、 
Ruby 和 Tcl 等 。 

(4) 支持 多 线程 ， 充 分 利用 CPU 资源 。 

(5) 优化 的 SQL 查询 算法 ， 有 效 地 提高 查询 速度 。 

(6) 既 能 够 作为 一 个 单独 的 应 用 程序 应 用 在 客户 端 服 务 器 网 络 环境 中 , 也 能 够 作为 一 个 库 而 嵌入 
到 其 他 软件 中 提供 多 语言 支持 ， 常 见 的 编码 如 中 文 的 GB2312、BIG5， 日 文 的 Shift_ JIS 等 都 可 以 用 作 
数据 表 名 和 数据 列 名 。 

(7) 提供 TCPHIP、ODBC 和 JDBC 等 多 种 数据 库 连 接 途 径 。 

(8) 提供 用 于 管理 、 检 查 、 优 化 数据 库 操作 的 管理 工具 。 
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(9) 可 以 处 理 拥有 上 千 万 条 记录 的 大 型 数据 库 。 
目前 的 最 新 版 本 是 MySQL 5.6， 它 提供 了 一 组 专用 功能 集 ， 在 当今 现代 化 、 多 功能 处 理 硬 件 和 软 
件 以 及 中 间 件 构架 涌现 的 环境 中 ， 极 大 地 提高 了 MySQL 的 性 能 、 可 扩展 性 、 可 用 性 。 
MySQL 5.6 融合 了 MySQL 数据 库 和 InnoDB 存储 引擎 的 优点 , 能 够 提供 高 性 能 的 数据 管理 解决 方 
包括 以 下 几 点 。 
(1) InnoDB 作为 默认 的 数据 库存 储 引擎 。 
(2) 提升 了 Windows 系统 下 的 系统 性 能 和 可 扩展 性 。 
(3) 改善 性 能 和 可 扩展 性 ， 全 面 利 用 各 平台 现代 多 核 构架 的 计算 能 
(4) 提高 实用 性 。 
(5) 提高 易 管 理性 和 效率 。 
(6) 提高 可 用 性 。 
(7) 改善 检测 与 诊断 性 能 。 


案 


2.3 MySQL 的 应 用 环境 


MySQL 与 其 他 大 型 数据 库 (如 Oracle、DB2、SQL Server 等 ) 相 比 ， 确 有 不 足 之 处 ， 如 规模 小 、 
功能 有 限 等 ， 但 是 这 丝毫 也 没有 减少 它 受 欢迎 的 程度 。 对 于 个 人 使 用 者 和 中 小 型 企业 来 说 ，MySQL 提 
供 的 功能 已 经 绰绰有余 ， 而 且 由 于 MySQL 是 开放 源 代码 软件 ， 因 此 可 以 大 大 降低 总 体 拥有 成 本 。 

目前 Intemet 上 流行 的 网 站 构架 方式 是 LAMP (Linux+Apache+MySQL+PHP) ， 即 使 用 Linux 作为 
操作 系统 ，Apache 作为 Web 服务 器 ，MySQL 作为 数据 库 ，PHP 作为 服务 器 端 脚本 解释 器 。 由 于 这 4 
个 软件 都 是 免费 或 开放 源 代 码 软件 (FLOSS) ， 因 此 使 用 这 种 方式 不 用 花 一 分 钱 〈 除 人 工 成 本 ) 就 可 
以 建立 起 一 个 稳定 、 免 费 的 网 站 系统 。 


2.4 MySQL 服务 器 的 安装 和 配置 


MySQL 是 目前 最 为 流行 的 开放 源码 的 数据 库 ， 是 完全 网 络 化 的 跨 平 台 的 关系 型 数据 库 系统 ， 它 是 
由 MySQL AB 公司 开发 、 发 布 并 支持 的 。 任 何人 都 能 从 Intemet 上 下 载 MySQL 软件 ， 而 无 须 支 付 任 
何 费用 ， 并 且 “ 开 放 源 代码 ”意味 着 任何 人 都 可 以 使 用 和 修改 该 软件 ， 如 果 愿 意 ， 用 户 也 可 以 研究 源 
代码 并 进行 恰当 的 修改 ， 以 满足 自己 的 需求 ， 不 过 需要 注意 的 是 ， 这 种 “自由 ”是 有 范围 的 。 


2.4.1 MySQL 服务 器 下 载 


MySQL 服务 器 的 安装 包 可 以 到 Oracle 官网 (http:/www-.oracle.comyindex.html) 中 下 载 。 下 载 
MySQL 的 具体 步 又 如 下 。 
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(1) 在 浏览 器 的 地 址 栏 中 输入 URL 地 址 http://www.oracle.com/index.html, 进入 Oracle 官网 首页 ， 


将 鼠标 移动 到 Downloads 上 ， 将 显示 如 图 2.2 所 示 的 Downloads 子 菜单 。 


Teernet Explorer 














大 orade| Hardware and Sofware Engineered to Work Together - Windows, 








[oS 





4 [xP a 





OE earede co ndechn 





富 必 二天 辐 Oracle | Hardware and scfware Engineered.. 





Q 


Products ”solutions Downloads Store Support Training Partners About 


Sign IwRegister Help Country ~ Communities ~ lama. ~ Iwantto.. ~ Search 


ORACLE 


Popular 
Downloads 


Developer Toole 
SQL Developer 
IDeveloper and ADF 


Entorpnse Wanagomont Applicatone 


Orade Enlerprse Hanager 


Databaco 
Orade Database 





Orace Datanase 119 Express Orade Applcaton Testng Sule 











Java for Your 2 ee Enterprise Fack for Ecipse CR 

Computer » HetDeane (DE Ap 

Ee 单 击 该 超 链接 ons er Ne 
ET Prebutt Developer Hs 


Madleware 
orade Fusion Middlaware 


Crate Databese (ncuding Oracle WebL ogic 
Prebult Developer Senen) 
Ws Orade JRocN 

Orade SOA Sulle 

Seo Ml 





bp/ mem mre @ ect| RMR 有 





ss Suit 


PeopleSeft JD Edwards, Sievel 





OTN 











2.2 Oracle 官网 的 Downloads 子 菜单 


(2 
动 到 底 间 





hh， 如 图 2.3 所 示 。 


MySQL Cluster CGE 


ORACLE 


图 2.3 MySQL Downloads 页 面 


) 在 如 图 2.2 所 示 的 菜单 中 ， 单 击 MySQL 超 链接 ， 进 入 到 MySQL Downloads 页 





面 ， 将 页 面 滚 





(3) 单 击 Community (GPL) Downloads >> 超 链接 ， 进 入 到 MySQL Community Downloads 页 面 ， 





如 图 2.4 所 示 。 
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xj tre | cag 


Mysar ~ 2 


urce database 













yseLcom LE 
MySQL Community Downloads 


单 击 该 超 链接 


MYSQLG 


MySQL Enterprise Edition 





monity Server is the worid's most popular open source database MySQt Enterpnse Eduon novdes 
the man comprehensive set of 
advanced features and 
monagerment toob tor MySQL 


Dow 





MySQL Cluster Lesm More = 


ommioad hom Orace eDehvery » 
ms 





Juster (3 3 real bme open source transacponal catabane 


Dow 





wo 
MY5QL Cluster CGE 


MySQL Fabric 


MySa Cluster sa real- tme, 
MySOQt Fabrie provides s tamework for managing Hgh avaalabssty and Shardng. 


(vansctonal database desgned 
PP fo yahoo on ace to tn 


图 2.4 MySQL Community Downloads 页 面 


(4) 单 击 MySQL Community Server(GPL) 超 链接 ， 将 进入 到 Download MySQL Community Server 
页 面 ， 将 页 面 滚动 到 如 图 2.5 所 示 的 位 置 。 




















六 a | 国 MsQt = Domioed MrsQLCommuniy Sener | | 





MySQL Community Server 5.6.20 


pinto @ 选择 操作 系统 


Recommended Download: 


More Countries » 







Contact Us Online » 







@ 单 击 该 图 片 ， 下 载 完整 
版 的 MySQL Server 安装 包 


Related Pages: 











MySQL Installer 5.6 
ws for Windows 


AI MysQL Products. For All Windows Platforms- 
In One Package. 











Windowa (x86, 64-blt), MySQ| 


目 如 果 是 Windows 32 位 操作 系统 ， 单 
击 该 按钮 下 载 精简 版 的 MySQL 服务 器 
Windows (x06, 32-bit), MST Inataller 620 Er 

@ 如 果 是 Windows 64 位 操作 系统 ， 单 
击 该 按钮 下 载 精简 版 的 MySQL 服务 器 


Other Downloads: 







































I 至 [TELEE 三 
































图 2.5 Download MySQL Community Server 页 面 


(5) 根据 自己 的 操作 系统 来 选择 合适 的 安装 文件 ， 这 里 以 针对 Windows 32 位 操作 系统 的 完整 版 
MySQL Server 为 例 进行 介绍 ， 单 击 图 2.5 中 的 图 片 ， 将 进入 到 Download MySQL Installer 页 面 ， 在 该 
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页 面 中 ， 滚 动 到 如 图 2.6 所 示 的 位 置 。 











MySQL Conneaars 四 
Generally Available (GA) Releases 。 Development Releases 


other Downioads 


MySQL Installer 5.6.20 


Contact Sales 
Select Platform: Looking for previous GA 


versions? 







USA: +1-866-221-0634 
Canada: +1-856-221-0634 Microsofl Windows 





| Germany: +49 89 143 01280 
France: +39 1 57 60 83 57 Windows (x86, 32-bit), MSE installer 
Taly: +39 02 249 59 120 


用 | uk: +44 207 ssa 8447 osol554| Sonotre 


(mse rataller web semmerity 5.6.20.0.ms) 


" 





Japan: 0120-055556 windows (x86, 32-blt), MSI Installer 5.6.20 248.2M Downioad 
| china: 10800-; 


| India; 0008001005870 





0823 





{mye inetaller-commerity 5.6.20.0.m) 10791488e337295455872055 | Sonature 


More counues » @ We sg0est that you uo0 the MOS dedcumaand cmupc signatures to verly the integrity of the packages you 


Contact Us online » downioad. 



































2.6 ” Download MySQL Installer 页 面 


(6) 单 击 Download 按钮 ， 将 进入 到 如 图 2.7 所 示 的 Begin Your Download - mysql-installer- 
community-5.6.20.0.msi 页 面 。 









Archtves 





windows 





Yum Repository 





Community APT Repository 


















ES Begin Your Download - mysql-i 


MySQL on Windows 


nstaller-community-5.6.20.0.msi 





Login Now or Sign Up for a free account. 
MySQL Yum Repository 
An Oracle Web Account provides you with the following advantages: 


MySQL APT Repository 
» Fast access to MySQL software downloads 


MySQL community server Download technical White papers and Presentations 
eva 。 post messages in the MySQL Discussion Forums 
Report and track bugs in the MySQL bug system 
MySQL Fabric » Comment in the MySQL Documentation 
MySQL Utilities 
MysQL Workbench rr 
nUp» 


MySQL Proxy ot an Orecse Weeb ecceend 





MySQL Connectors 


Gi MySQL.com is using Oracle SSO for authentication. 1f you already have an Oracle Web 
ler Downloads 
account, click the Login link. Otherwise, you can signup for a free account by clicking the Sign 


| 
| 

| 

| Up link and following the instrucd 

| | 单 击 该 超 链接 


No thanks, just start my download. 


2.7 Begin Your Download - mysql-installer-community-5.6.20.0.msi 页 面 


(7) 单 击 No thanks, just start my download. 超 链接 ， 将 打开 如 图 2.8 所 示 的 文件 下 载 对 话 框 ， 单 击 
“保存 ”按钮 ， 下 载 安装 文件 。 
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您 星 要 运行 还 用 保存 来 后 cdn.mysql.com 的 mysql-installer-community-5.6.20.0.msi (248 MB)? 











2.4.2 MySQL 服务 器 安装 


下 载 MySQL 服务 器 的 安装 文件 以 后 , 将 得 到 一 个 名 称 为 mysql-installer-community-5.6.20.0.msi 的 
安装 文件 ， 双 击 该 文件 可 以 进行 MySQL 服务 器 的 安装 ， 具 体 的 安装 步骤 如 下 。 
(1) 双击 下 载 后 的 mysql-installer-community-5.6.20.0.msi 文件 ， 打 开 安装 向 导 ， 如 果 没 有 打开 安 
装 向 导 ， 而 是 弹出 如 图 2.9 所 示 的 对 话 框 ， 那 么 还 需要 先 安装 .NET 4.0 框架 ， 然 后 再 重新 安装 双击 下 载 
后 的 安装 文件 ， 打 开 安 装 向 导 对 话 框 ， 如 图 2.10 所 示 。 





ires NET Framework 4.0. Please 
4 eee ET er hen on We ceb te | 
For more Ini 


formation, please see 
http/igo.microsoft comyfwink?Linkid=181012 


Cs) 


图 2.9 打开 需要 安装 NET 4.0 框架 的 提示 对 话 框 








A 
Welcome MySQL 


The MySQL Instaler gudes you through the nstalaton and Ca 
MySQL products. Run it from the Start Menu to perform maintenance tasks later. 





Select one of the actions balow: 


要 


netal souProduds 
Guide you through the matalason ord confguraton of your 
MysQ producis. 


rn ehcenty on "| 























2.10 ”安装 向 导 对 话 框 


(2) 在 打开 的 安装 向 导 对 话 框 中 单 击 Install MySQL Products 超 链接 ， 将 打开 License Agreement 
对 话 框 ， 询 问 是 否 接受 协议 ， 选 中 Iaccept the license terms 复 选 框 ， 接 受 协议 ， 如 图 2.11 所 示 。 
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Uicense Agreement 

















Ce Ce Ce 








图 2.11 License Agreement 对 话 框 


(3) 单 击 Next 按钮 ， 将 打开 Find latest products 对 话 框 。 在 该 对 话 框 中 ， 选 中 Skip the check for 
updates(not recommended) 复 选 框 ， 这 时 ， 原 来 的 Execute 按钮 将 转换 为 Next 按钮 ， 如 图 2.12 所 示 。 
[DO wo sealer 
eo 1 Find latest products 


Before the inctalation 上 | the Inctaller will check if there are newer 
versions of the products you are about to install / aready installed are avallable. 


Medlavestprocucs 


I Sho the dhed for updates (not recommended) 





2.12 ”Find latest products 对 话 框 
(4) 语 





和 击 Next 按钮 ,将 打开 Choosing a Setup Type 对 话 框 ， 在 该 对 话 框 中 ， 共 包括 Developer Default 
(开发 者 默认 ) 、Server only( 仅 服务 器 ) 、Client only 〈 仅 客户 端 ) 、Full (完全 ) 和 Custom (〈 自 定 


义 ) 5 种 安装 类 型 ， 这 里 选择 Developer Default， 并 且 将 安装 路 径 修 改 为 “C:\Program Files\MySQI\”， 
数据 存放 路 径 修改 为 “C:\ProgramDatakMySQL\IMySQL Server 5.6\”， 如 图 2.13 所 示 。 
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一 一 
区 —— ee 
Ee Choosing a Setup Type 
MySQL. Installer || 
Please select the Setup Type that suits your use case. 
® Developer Default | 
nstals al products needed for EE 
- teasred fp MYSQL appkcaton develooment 
MYSQL development purposes. [This ‘5 usefid if you ntend to develop 
odcabons for an xeng server 引 | 
B Server only [This Setup Type ndudes: 
Instals oly he MysQ Server a 
pr 
Chent only 
Instals only the MySQL Chent 
Predict Wathout a server 图 
Full natalaton Path ~- 用 
Instals al melded MysQL prodauts EProrm es Wea 国 
nd eahres 
Custom Data Path 
Manualy select the products that EProg arOata WySc WyS ey 











‘shod be rstabed on the system 


Ce | Ce Lee |] 




















图 2.13 ”Choosing a Setup Type 对 话 框 

(5) 单 击 Next 按钮 ， 将 打开 如 图 2.14 所 示 的 Check Requirements 对 话 框 ， 在 该 对 话 框 中 检查 系 

统 是 否 具备 安装 所 必需 的 .NET 4.0 框架 和 Microsoft Visual C++ 2010 32-bit runtime， 如 果 不 存在 ， 单 击 
Execute 按钮 ， 将 在 线 安装 所 需 插 件 ， 安 装 完成 后 ， 将 显示 如 图 2.15 所 示 的 对 话 框 。 

WsQL instaler TT 










AN Check Requirements 
MySQL. Installer 中 
The following requirements must be installed before the selected products can be 
installed. Tf you don't want a particular requirement then go back and deselect the 
product that requires it. 





Requrement For Product status 
@ Microsoft NET Framework 4 Cliest Profile MySQL Notifier 1.1.5 

© Microsoft Visual C++ 2010 32-bit runtime MySQL Workbench CE 6.1.7 

O Microsoft NET Framework4 Chient Proflle MysQL Workbench CE 6.1.7 





Check Requirements 




















图 2.14 未 满足 全 部 安装 条 件 时 的 Check Requirements 对 话 框 
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Nor Installer 









Microsoft .NET Framework 4 client Profile MySQL Notifiler 1.1.5 四 
加 Microsoft Visual C++ 2010 32-bit runtime MySQL Workbench CE647 
© Microsoft .NET Framework 4 Client Profile MySQL Workbench CE 6.1.7 











Current Task 
A required prerequisites are met Continue by didang on the Next button. 


Ce | Cu | or |] 











图 2.15 ”安装 条 件 已 全 部 满足 时 的 Check Requirements 对 话 框 
(6) 单 击 Next 按钮 ， 将 打开 如 图 2.16 所 示 的 Installation Progress 对 话 框 。 








© wsat installer 本: 
Installation Progress 
Q e 
The folowing products wal be installed or updated. 
Product Status Progress Notes 
加 wsesrmessz To beinstalled 
EE] wser wortoenc ces17 To beinstalled 
区 wser owner 11s To beinstalled 
四 weer veiwes 143 To be nstalled 
Ee) comecoro0ecs34 To be instaled 
mm ©) comecorcrr 113 To beinstalled 
园 <meaomsaai Tobenstalled 
Ea) cmeaomersss To be nstalled 
园 wset comecorcs1s To beinstalled 
EE] wsar pocmentason ss20 To beinstalled 
EE] samoles nd amples ss20 To beinstalled i 
Chick Eveaute] to nstal or update the folowng padiages | 














图 2.16 未 安装 完成 的 Installation Progress 对 话 框 
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(7) 单 击 Execute 按钮 ， 将 开始 安装 ， 并 显示 安装 进度 。 安 装 完成 后 ， 将 显示 如 图 2.17 所 示 的 对 
话 框 。 


DN Installation 
MysQL Installer ED OU 


The folowing products wil be installed or updated. 





Product Staus 


区 ] maser sererss2 mstall sucess 





区 | wsor workbena ces23s nstall suceess 





®| comeconopsc5110 nstall suctess 





mecor/c++ 3.4.0 netall ruccecs 





mecor/c 60.2 nstall suctess 
mecor S19 nstall suceess 


ComecorNET 6.44 nstall suctess 





MysQt Documentation 55.24 mstall suceess 





加 
© 
加 
已 
加 
© 
加 
加 上 





mples and Beanples 55.24 nstall suceess 





2.17 ”安装 完成 时 的 Installation Progress 对 话 框 


(8) 单 击 Next 按钮 ， 将 打开 Configuration Overview 对 话 框 ， 在 该 对 话 框 中 单 击 Next 按钮 ， 将 打 
开 用 于 选择 服务 器 的 类 型 的 MySQL Server Configuration 对 话 框 ， 在 该 对 话 框 中 共 提 供 了 Development 
Machine (开发 者 类 型 ) 、Server Machine (服务 器 类 型 ) 和 Dedicated Machine (致力 于 MySQL 服务 类 
型 ) 。 这 里 选择 默认 的 Development Machine， 如 图 2.18 所 示 。 
Ma mi TI EL 一 | 
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芯 ， 
MySQL Installer 


Server Configuration Type 



































图 2.18 配置 服务 器 类 型 和 网 络 选项 的 对 话 框 
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。 MySQL 使 用 的 默认 端口 是 3306, 在 安装 时 ,可 以 修改 为 其 他 的 ( 如 3307)。 但 是 一 般 情 况 下 ， 
， 不 要 修改 默认 的 端口 号 ， 除 非 3306 端口 已 经 被 占用 。 

(9) 单 击 Next 按钮 ， 将 打开 用 于 设置 用 户 和 安全 的 MySQL Server Configuration 对 话 框 ， 在 这 个 
对 话 框 中 , 可 以 设置 root 用 户 的 登录 密码 , 也 可 以 添加 新 用 户 , 这 里 只 设置 root 用 户 的 登录 密码 为 root， 
其 他 采用 默认 ， 如 图 2.19 所 示 。 














图 2.19 设置 用 户 和 安全 的 MySQL Server Configuration 对 话 框 


(10) 单 击 Next 按钮 ， 将 打开 Configuration Overview 对 话 框 ， 开 始 配置 MySQL 服务 器 ， 这 里 采 
用 默认 设置 ， 如 图 2.20 所 示 。 








2.20 配置 MySQL 服务 器 
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(11) 单 击 Next 按钮 ， 将 显示 如 图 2.21 所 示 的 界面 ， 提 示 安 装 MySQL 提供 的 简单 示例 。 








MySQL Installer [= le 
/SE i 
RS Configuration Overview 
MySQL. Installer 
The following products wil now be configured. 
Product Acton to be performed Progress | 
厂 
号 EE] smple and amples 5.620 Inital Configuration, 
| 
| 
Show Detals > 
Es 医 瑟 5 

















图 2.21 提示 安装 MySQL 提供 的 简单 示例 
(12) 单 击 Next 按钮 ， 开 始 安装 ， 安 装 完成 后 ， 将 显示 如 图 2.22 所 示 的 界面 。 

















Mo - ETC 
NN Configuration Overview 
MySQL. Installer 
The following products will now be configured. 
Product Acton to be performed Progress | 
v 
v 
| 
| 
how Detas > | 
| 

















图 2.22 配置 完成 界面 
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(13) 单 击 Next 按 钮 ,将 显示 如 图 2.23 所 示 的 安装 完成 界面 .取消 选中 Start MySQL Workbench after 
Setup 复 选 框 ， 单 击 Finish 按钮 ， 完 成 MySQL 的 安装 。 


癌 MysQL Installer 


Installation Complete 


a Installer 


The installation procedure has been completed. 


贺 start MySQL Workbench after Setup 


Complete 

















图 2.23 ”安装 完成 对 话 框 
2.4.3 ”启动 、 连 接 、 断 开 和 停止 MySQL 服务 器 


通过 系统 服务 器 和 命令 提示 符 (DOS) 都 可 以 启动 、 连 接 断 开 和 停 击 MySQL， 操 作 非 常 简单 。 下 
面 以 Windows 7 操作 系统 为 例 ， 讲 解 其 具体 的 操作 流程 。 通 常情 况 下 不 要 停止 MySQL 服务 器 ， 和 否则 
数据 库 将 无 法 使 用 。 


1. 启动 、 停 止 MySQL 服务 器 


启动 、 停 止 MySQL 服务 器 的 方法 有 两 种 : 系统 服务 器 和 命令 提示 符 (DOS) 。 

1) 通过 系统 服务 器 启动 、 停 止 MySQL 服务 器 

如 果 MySQL 设置 为 Windows 服务 ， 则 可 以 通过 选择 “开始 ”一 “控制 面板 ”一 “系统 和 安全 ” 
“管理 工具 ”一 “服务 ”命令 打开 Windows 服务 管理 器 。 在 服务 器 的 列表 中 找到 MySQL 服务 并 右 
单 击 , 在 弹出 的 快捷 菜单 中 , 完成 MySQL 服务 的 各 种 操作 〈 启 动 、 重 新 启动 、 停 止 、 暂 停 和 恢复 ) ， 
图 2.24 所 示 。 























键 
如 
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可 = | 
文件 (月 ” 报 作 (A) ”得 看 (V) 帮助 (H) 
外 人 宁 | 国 日 喇 ml 
服务 (本 地 ) 




















Netpi 停止 (0) 华 用 本 地 服务 
| NetT( 和 暂 壤 U) 本 用 本 地 服务 
| NetTl 复 人 全 用 本 地 服务 

Nedlo| 。 重新 启动 (E) 手动 本 地 系统 

Netwq 所 有 任务 (Q » 中 

Netwc 

U) 

Netwc 属性 (R) 

Netwd 

启动 
了 帮助 重新 启动 ) 


四 | 扩展 人 际 全 
| 停止 并 启动 本 地 计算 机 上 的 服务 MySQL56 




















图 2.24 通过 系统 服务 启动 、 停 止 MySQL 服务 器 
2) 在 命令 提示 符 下 启动 、 停 止 MySQL 服务 器 
单 击 “ 开 始 ” 菜 单 ， 在 出 现 的 命令 输入 框 中 输入 cmd 命令 ， 按 Enter 键 打 开 DOS 窗 [ 
示 符 下 输入 : 
\> net start mysql 
此 时 再 按 Enter 键 ， 启 用 MySQL 服务 器 。 
在 命令 提示 符 下 输入 : 


\> net stop mysql 





。 在 命令 提 


按 Enter 键 ， 即 可 停止 MySQL 服务 器 。 在 命令 提示 符 下 启动 、 停 止 MySQL 服务 器 的 运行 效果 如 


图 2.25 所 示 。 





国 管理 员 : C\Windows\system32\cmd.exe je 


> 2889 Microsoft Corporation 


Users\Adninistratormnet start nysql 
wseL 
ysQL 








图 2.25 在 命令 提示 符 下 启动 、 停 止 MySQL 服务 器 
2. 连接 和 断 开 MySQL 服务 器 


下 面 分 别 介绍 连接 和 断 开 MySQL 服务 器 的 方法 。 
1) 连接 MySQL 服务 器 
连接 MySQL 服务 器 通过 mysql 命令 实现 。 在 MySQL 服务 器 启动 后 ， 选 择 “开始 ”一 
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1 


运行 ” 命 
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令 ， 在 弹出 的 “运行 ”窗口 中 输入 cmd 命令 ， 按 Enter 键 后 进入 DOS 窗口 ， 在 命令 提示 符 下 输入 : 
\>mysql -uroot —h127.0.0.1 —p password 


MySQL 服务 用 
器 所 在 地 址 


疏 卫 型 


密 
码 
a 
SB 注意 
在 连接 MySQL 服务 器 时 ，MySQL 服务 器 所 在 地 址 (如 -hl127.0.0.1 ) 可 以 省 略 不 写 。 


输入 完 命令 语句 后 ， 按 Enter 键 即 可 连接 MySQL 服务 器 ， 如 图 2.26 所 示 。 
| 








TT 











图 2.26 连接 MySQL 服务 器 


位 明 
为 了 保护 MySQL 数据 库 的 密码 ， 可 以 采用 如 图 2.26 所 示 的 密码 输入 方式 。 如 果 密 码 在 卫 后 
直接 给 出 ， 那 么 密码 就 以 明文 显示 ， 例 如 : 
mysql -uroot -hl27.0.0.1 -proot 
按 Enter 键 后 再 输入 密码 ( 以 加 密 的 方式 显示 ) ， 然 后 按 Enter 键 即 可 成 功 连接 MySQL 服务 器 。 


如 果 用 户 在 使 用 mysql 命令 连接 MySQL 服务 器 时 弹出 如 图 2.27 所 示 的 信息 ， 那 么 说 明 用 户 未 设 
置 系统 的 环境 变量 。 











图 2.27 连接 MySQL 服务 器 出 错 
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也 就 是 说 ， 没 有 将 MySQL 服务 器 的 bin 文件 夹 位 置 添加 到 Windows 的 “环境 变量 ”一 “系统 变 
量 ” 一 Path 中 ， 从 而 导致 命令 不 能 执行 。 

下 面 介绍 这 个 环境 变量 的 设置 方法 ， 其 步骤 如 下 。 

(1) 右键 单 击 “ 计 算 机 ”图 标 ， 在 弹出 的 快捷 菜单 中 选择 “属性 ”命令 ， 在 弹出 的 对 话 框 中 选择 
“高 级 系统 设置 ”， 弹 出 “系统 属性 ”对 话 框 ， 如 图 2.28 所 示 。 

(2) 在 “系统 属性 ”对 话 框 中 ， 选 择 “ 高 级 ”选项 卡 ， 单 击 “ 环 境 变 量 ” 按 钮 ， 弹 出 “环境 变量 ” 
对 话 框 ， 如 图 2.29 所 示 。 





























”村 环境 县 [Ex]) 
计 笠 机 各 |[ 蔬 件 “| 高 级 | 未 纺 保护 [运程 | Ca 
要 进行 大 多 数理 下 ， 您 必须 作为 管理 员 登 录 。 Administrator 的 用 户 变量 如 
性 能 到 时 值 
视 六 效果 ， 处 理 吕 计 划 ， 内 存 使 用 ， 以 及 虚拟 内 存 Ph 
TENP KUSERPROFILEX\AppData\Locsl\Tenp 
THP KUSERPROFILEX\AppData\Local\Temp 
用 户 轩 文件 











与 您 登录 有 关 的 点 面 设置 新 嫂 如 ..，| | 编辑 EE)..，| | 量 除 钾 ) 





启动 和 地 障 恢复 
系统 启动 、 系 统 失 败 和 调试 信息 














[2] 





[CC 三 


























图 2.28 “系统 属性 ”对 话 框 图 2.29 “环境 变量 ”对 话 框 


(3) 在 “环境 变量 ”对 话 框 中 ， 定 位 到 “系统 变量 ”中 的 Path 选项 ， 单 击 “ 编 辑 ” 按 钮 ， 将 弹 
出 “编辑 系统 变量 ”对 话 框 ， 如 图 2.30 所 示 。 








变量 名 中 Path 


变量 值 中: 

















图 2.30 “编辑 系统 变量 ”对 话 框 


(4) 在 “编辑 系统 变量 ”对 话 框 中 ， 将 MySQL 服务 器 的 bin 文件 夹 位 置 (C:\Program 
FilesIMySQIL\MySQL Server 5.6\bin〉 添加 到 “变量 值 ”文本 框 中 ， 注 意 要 使 用 “:” 与 其 他 变量 值 进行 
分 隔 ， 最 后 ， 单 击 “ 确 定 ” 按 钮 。 

环境 变量 设置 完成 后 ， 再 使 用 mysql 命令 即 可 成 功 连接 MySQL 服务 器 。 
2) 断 开 MySQL 服务 器 
连接 到 MySQL 服务 器 后 , 可 以 通过 在 MySQL 提示 符 下 输入 exit 或 者 quit 命令 断 开 MySQL 连接 ， 
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格式 如 下 。 


mysql> quit; 
2.4.4 打开 MySQL 5.6 Command Line Client 


MySQL 服务 器 安装 完成 后 ， 就 可 以 通过 其 提供 的 MySQL 5.6 Command Line Client 程序 来 操作 
MySQL 数据 了 。 这 时 ， 必 须 先 打开 MySQL 5.6 Command Line Client 程序 ， 并 登录 MySQL 服务 器 。 
下 面 将 介绍 具体 的 步骤 。 
(1) 在 “开始 ”菜单 中 ， 选 择 “ 所 有 程序 ”一 MySQL 一 MySQL Server 5.6 一 MySQL 5.6 Command 
Line Client 命令 ， 将 打开 MySQL 5.6 Command Line Client 窗口 ， 如 图 2.31 所 示 。 


国 MysQL 56 Command Line Client : ey | 


























图 2.31 MySQL 客户 端 命令 行 窗 口 





(2) 在 该 窗口 中 , 输入 root 用 户 的 密码 (这 里 为 root) ， 将 登录 到 MySQL 服务 器 ， 如 图 2.32 所 示 。 





站 
国 wysat 56 command Line Client | 





| © lc) 
Comnands end with ; or \g. 
Dommunity Server 《GPL》 
s affiliates. All rights reserved. 


le Corporation and/or its 
of their respective 


Type ’help;’ or ’\h’ for help- '\c’ to clear the current 


input statement. 


nysql> - 

















图 2.32 登录 到 MySQL 服务 器 
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2.$ ”如何 学 好 MySQL 


学 好 MySQL 最 重要 的 是 要 多 练习 。 笔 者 将 自己 学 习 数据 库 的 方法 总 结 如 下 。 

1. 多 上 机 实践 

要 想 熟 练 地 掌握 数据 库 ， 必 须 经 常 上 机 练习 。 只 有 在 上 机 实践 中 才能 深刻 体会 数据 库 的 使 用 。 通 
常情 况 下 ， 数 据 库 管 理 员 工作 的 时 间 越 长 ， 其 工作 经 验 就 越 丰 富 。 很 多 复杂 的 问题 ， 都 可 以 根据 数据 
库 管 理 员 的 经 验 来 更 好 地 解决 。 上 机 实践 的 过 程 中 ， 可 以 将 学 到 的 数据 库 理论 知识 理解 得 更 加 透彻 。 

2. 多 编写 SQL 语句 

SQL 语句 是 数据 库 的 灵魂 。 数 据 库 中 的 很 多 操作 都 是 通过 SQL 语句 来 实现 的 。 只 有 经 常 使 用 SQL 
语句 来 操作 数据 库 中 的 数据 ， 读 者 才 可 以 更 加 深刻 地 理解 数据 库 。 

3. 数据 库 理论 知识 不 能 丢 

数据 库 理论 知识 是 学 好 数据 库 的 基础 。 虽 然 理论 知识 会 有 点 儿 枯燥 ,但 是 这 是 学 好 数据 库 的 前 提 。 
例如 ， 数 据 库 理论 中 会 涉及 E-R 图 、 数 据 库 设计 原则 等 知识 。 如 果 不 了 解 这 些 知识 ， 就 很 难 独立 设计 
一 个 很 好 的 数据 库 及 表 。 读 者 可 以 将 数据 库 理论 知识 与 上 机 实践 结合 到 一 起 来 学 习 ， 这 样 效率 会 提高 。 





2.6° 大 结 


本 章 介绍 了 数据 库 和 MySQL 的 基础 知识 。 通 过 本 章 的 学 习 ， 希 望 读者 对 数据 库 、MySQL 数据 库 
和 SQL 等 知识 有 所 了 解 。 而 且 ， 希 望 读 者 能 够 了 解 常用 的 数据 库 系统 。 第 3 章 将 介绍 在 Windows 操作 
系统 下 安装 和 配置 MySQL， 同 时 对 数据 库 的 相关 知识 也 要 有 一 定 的 了 解 。 


2.7 实践 与 练习 


1. 到 Oracle 的 官网 中 下 载 最 新 版 本 的 MySQL 服务 器 并 安装 。 
2. 尝试 在 命令 提示 符 下 启动 、 停 止 MySQL 服务 器 。 
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使 用 MySQL 图 形 化 管理 工具 


( 敬 视频 讲解 : 2S 分 钟 ) 


My5QL 的 管理 维护 工具 非常 多 ， 除 了 系统 自 带 的 命令 行 管理 工具 之 外 ， 还 有 
许多 其 他 的 图 形 化 管理 工具 ， 常 用 的 有 MySQL Workbench、phpMyAdmin、 
Navicat 等 。 通 过 这 些 第 三 方 的 管理 工具 ， 可 以 使 MySQL 的 管理 更 加 方便 。 本 章 
将 对 MySQL Workbench 图 形 化 管理 工具 和 phpMyAdmin 图 形 化 管理 工具 进行 系 


统 讲 全。 


于 于 于 于 于 于 于 于 至 于 至 


过 阅读 本 章 ， 读 者 可 以 : 
了 解 MySQL Workbench 图 形 化 管理 工具 
通过 MySQL Workbench 图 形 化 管理 工具 创建 数据 库 和 数据 表 
通过 MySQL Workbench 图 形 化 管理 工具 添加 数据 
通过 MySQL Workbench 图 形 化 管理 工具 导出 导入 数据 
配置 phpMyAdmin 图 形 化 管理 工具 
使 用 phpMyAdmin 图 形 化 管理 工具 实现 数据 库 与 数据 表 的 创建 
通过 phpMyAdmin 图 形 化 管理 工具 对 数据 库 与 数据 表 进行 管理 
使 用 phpMyAdmin 图 形 化 管理 工具 对 数据 库 执行 导入 导出 
通过 phpMyAdmin 图 形 化 管理 工具 设置 数据 的 编码 格式 
通过 phpMyAdmin 图 形 化 管理 工具 添加 服务 中 新 用 户 
通过 phpMyAdmin 图 形 化 管理 工具 重 置 MySQL 服务 路 用 户 密码 
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3.1 MySQL Workbench 图 形 化 管理 工具 


MySQL Workbench 是 MySQL AB 发 布 的 可 视 化 的 数据 库 设 计 软 件 ， 它 的 前 身 是 FabForce 公司 的 
DB Designer 4。MySQL Workbench 是 为 数据 库 管 理 员 、 程 序 开 发 者 和 系统 规划 师 设 计 的 统一 的 可 视 化 
工具 。 它 提供 了 先进 的 数据 建 模 , 灵活 的 SQL 编辑 器 和 全 面 的 管理 工具 , 可 在 Windows、Linux 和 Mac 
等 操作 系统 上 使 用 。 

数据 建 模 一 一 MySQL Workbench 包括 所 有 数据 建 模 工程 需要 的 功能 ， 能 正 向 和 反 向 建立 复杂 的 
ER 模型 ， 也 提供 了 通常 需要 花 更 多 时 间 才 能 完成 的 变更 管理 和 文档 任务 的 关键 功能 。 

SQL 编辑 器 一 一 MySQL Workbench 提供 了 用 于 创建 、 执 行 和 优化 SQL 查询 的 可 视 化 工具 。SQL 
编辑 器 提供 了 语法 高 亮 显示 ，SQL 代码 复 用 和 执行 的 SQL 历史 。 数据库 的 连接 面板 允许 开发 人 员 轻 松 
地 管理 数据 库 连 接 。 对 象 浏览 器 提供 即时 访问 数据 库 模型 和 对 象 。 

管理 工具 一 一 MySQL Workbench 提供 了 可 视 化 的 控制 台 ， 能 轻松 管理 MySQL 数据 库 环 境 ， 并 为 
数据 库 增 加 了 更 好 的 可 视 性 。 开 发 人 员 和 DBA 可 以 使 用 可 视 化 工具 配置 服务 器 , 管理 用 户 和 监控 数据 
库 的 健康 状况 。 


3.1.1 了 解 MySQL Workbench 
MySQL 数据 库 安装 完成 后 ， 将 自动 安装 一 个 图 形 化 工具 ， 用 于 创建 并 管理 数据 库 。 在 “开始 ” 菜 


单 中 选择 “所 有 程序 ”一 MySQL 一 MySQL Workbench 6.1 CE 命令 ,将 打开 如 图 3.1 所 示 的 MySQL 
Workbench 主 界面 。 





国 wsot worbed oi [ET 一 
本 5 


单 击 这 个 按钮 可 以 创建 一 
个 新 的 本 地 服务 器 连接 


单 击 这 个 按钮 管理 本 地 服 
务 器 的 连接 


默认 创建 的 本 地 服务 器 连 
接 ， 单 击 它 可 以 进入 到 
SQL 编辑 和 管理 界面 
单 击 下 面 的 超 链接 可 以 
打开 相应 的 功能 





图 3.1 MySQL Workbench 主 界面 


在 图 3.1 中 ， 单 击 Local instance MySQL56 超 链接 ， 将 打开 一 个 输入 用 户 密码 的 对 话 框 ， 在 该 对 话 
框 中 输入 root 用 户 的 密码 ， 这 里 为 root， 如 图 3.2 所 示 。 
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人 1 Please enter password for the 
Ne [i following service: 
Workbench, 


回 Save password in vault 


Co Co |] 








图 3.2 输入 用 户 密码 对 话 框 


单 击 OK 按钮 ， 将 打开 如 图 3.3 所 示 的 Local instance MySQL56 选项 卡 。 在 该 选项 卡 中 ， 可 以 进行 
创建 /管理 数据 库 、 创 建 /管理 数据 表 、 编 辑 表 数 据 和 查询 表 数 据 等 操作 。 











3.3 ”Local instance MySQL56 选项 卡 


3.1.2 创建 数据 库 和 数据 表 


下 面 将 介绍 如 何 应 用 MySQL Workbench 图 形 化 管理 工具 创建 数据 库 和 数据 表 。 
1. 创建 数据 库 


通过 MySQL Workbench 创建 数据 库 的 具体 方法 如 下 。 
(1) 打开 MySQL Workbench 的 Local instance MySQL56 选项 卡 ， 单 击 工具 栏 中 的 “创建 数据 库 ” 
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按钮 天 ， 将 打开 如 图 3.4 所 示 的 新 建 数据 库 选 项 卡 。 





图 3.4 新 建 数据 库 选项 卡 


Bo 
本名 曙 
在 MySQL 中 , Create Schema 和 Create Database 的 作用 是 一 样 的 , 都 是 创建 数据 库 。 在 MySQL 
Workbench 中 ， 创 建 数据 库 使 用 的 是 Create Schema。 


(2) 在 如 图 3.4 所 示 的 新 建 数据 库 选项 卡 中 ， 在 Name 文本 框 中 输入 数据 库 名 称 。 这 里 的 数据 库 
名 称 必 须 符 合 操作 系统 文件 夹 的 命名 规则 ， 而 在 MySQL 中 是 不 区 分 大 小 写 的 。 这 里 创建 一 个 名 称 为 
db_database02 的 数据 库 。 

(3) 在 Collation 下 拉 列 表 框 中 选择 所 创建 数据 库 的 字符 集 校对 规则 。 例 如 ， 要 使 用 UTF8 编码 ， 
则 可 以 选择 utf8 - utf8_general ci， 也 可 以 选择 utf8 - default collation。 这 里 选择 utf8 - default collation， 
如 图 3.5 所 示 。 





Rename References 


Colaton: [utfa-defoult colaton 





图 3.5 创建 数据 库 db_database02 
(4) 单 击 Apply 按钮 ， 将 弹出 如 图 3.6 所 示 的 对 话 框 ， 显 示 生 成 的 可 编辑 的 创建 数据 库 的 SQL 语句 。 
Apphy SQL Scriptto Dateboce 要 6 和 06 加 


Review the SQL Script to be Applied on the Database 








更 cREATE SCMEMA “db_database02” DEFAULT CHARACTER SET utfe ; 
2 














图 3.6 可 编辑 的 创建 数据 库 的 SQL 语句 对 话 框 
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(5) 单 击 Apply 按钮 ， 开 始 创 建 数据 库 。 数 据 库 创 建 完成 后 ， 显 示 如 图 3.7 所 示 的 完成 对 话 框 。 
(6) 单 击 Finish 按钮 ， 关 闭 完 成 对 话 框 。 
创建 数据 库 后 ， 在 Local instance MySQL56 选项 卡 的 左 侧 的 SCHEMAS 列表 中 ， 将 显示 新 创建 的 
数据 库 ， 如 图 3.8 所 示 。 
AppysQtseipt'oDetcbec sl 眉 | 








Applying SQL seript to the database 
Apply Sat scnpt 


The folowng tasks wil row be exeautsd PEase monitor he executon 
Press Show logs t see the executon logs, 


Ereaste SAL Satenents 


So saipt was successfully appled to the datapase. 









































db_databas ea 
图 3.7 创建 数据 库 完成 对 话 框 图 3.8 新 创建 的 数据 库 
2. 创建 数据 表 


通常 情况 下 ， 创 建 数据 库 后 ， 还 需要 创建 数据 表 。 在 MySQL Workbench 中 ， 创 建 数据 表 前 ， 需 要 
先 选 择 要 创建 数据 表 的 那个 数据 库 ， 这 时 可 以 在 Local instance MySQL56 选项 卡 的 左 侧 的 SCHEMAS 
列表 中 ， 双 击 该 数据 表 来 将 其 设置 为 默认 数据 库 ， 然 后 就 可 创建 数据 表 了 。 通 过 MySQL Workbench 
创建 数据 表 的 具体 方法 如 下 。 

(1) 打开 MySQL Workbench 的 Local instance MySQL56 选项 卡 ， 单 击 工具 栏 中 的 “创建 数据 表 ” 
按钮 天 ， 将 打开 如 图 3.9 所 示 的 新 建 表 选项 卡 。 














图 3.9 新 建 表 选 项 卡 
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(2) 在 新 建 表 选 项 卡 的 Table Name 文本 框 中 ， 输 入 数据 表 名 称 〈 这 里 为 tb_user) ; 在 Collation 
下 拉 列 表 框 中 选择 字符 集 较 对 规则 为 utf8 - default collation; 在 Engine 下 拉 列 表 框 中 选择 存储 引擎 为 


MyISAM， 如 图 3.10 所 示 。 























图 3.10 ”指定 数据 表 名 称 、 字 符 集 校对 规则 以 及 存储 引擎 
(3) 在 如 图 3.9 所 示 的 添加 字段 列表 中 ， 添 加 如 图 3.11 所 示 的 3 条 字段 信息 




































































Column Name Datatype PK NN UQ BIN UN ZF AL Defaukt 
hid INT 加 图 回回 回回 图 
> name VARCHAR(45) 团 四 四 四 EE 
9 pwd VARCHAR(45) 回回 固 固 回回 四 
= PF 闫 辣 必 
图 3.11 添加 字段 列表 











(4) 单 击 图 3.9 中 的 Apply 按钮 ， 将 弹出 如 图 3.12 所 示 的 对 话 框 ， 显 示 生 成 的 可 编辑 的 创建 数据 


表 的 SQL 语句 。 


Review the SQL Script to be Applied on the Database 


Review SQL Script 


CREATE TABLE 'db_database02'.tb_user ( 
id” INT NOT NULL AUTO_INCREMENT, 
N 











图 3.12 可 编辑 的 创建 数据 表 的 SQL 语句 
(5) 单 击 Apply 按钮 ， 开 始 创 建 数据 表 。 数 据 表 创建 完成 后 ， 显 示 如 图 3.13 所 示 的 完成 对 


话 框 。 
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Applying SQL script to the database … 


Apply SQt script 
The folowing tasks wl now be executed. Please monitor the execution. 
Press Show Logs to see the exeaution logs. 
Execute SQL Statements 


SQL saript was successfuly appied to the database. 





图 3.13 ”创建 数据 表 完 成 对 话 框 
(6) 单 击 Finish 按钮 ， 关 闭 完成 对 话 框 。 


3.1.3 ”添加 数据 


数据 库 和 数据 表 创 建成 功 后 ， 就 可 以 向 数据 表 中 添加 数据 了 。 在 MySQL Workbench 的 Local 
instance MySQL56 选项 卡 中 ， 向 已 有 数据 表 中 添加 数据 的 具体 步骤 如 下 。 

(1) 在 Local instance MySQL56 选项 卡 的 左 侧 的 SCHEMAS 列表 中 ,找到 要 添加 数据 的 数据 表 节 
点 ,这 里 为 tb_user 节点 ， 并 且 在 该 节点 上 单 击 鼠标 右键 ,在 弹出 的 快捷 菜单 中 选择 Select Rows - Limit 


5000 命令 ， 如 图 3.14 所 示 。 
SCHEMAS Ca 
Q Filter objects 


了 目 db_databa 
vB Tables 


* Bh Fundi 


如 ter Table 
Table maintenance .. 
Drop Table. .. 
Truncate Table... 


Search Table Data .. 
Refresh All 








图 3.14 找到 要 添加 数据 的 数据 表 tb_user 
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了 技 瑟 
在 如 图 3.14 所 示 的 快捷 菜单 中 ， 不 但 可 以 执行 数据 的 添加 命令 ， 还 可 以 执行 查询 数据 、 复 制 
数据 、 修 改 表 结构 和 删除 表格 等 操作 。 


(2) 这 时 将 打开 一 个 名 为 tb_user 的 选项 卡 ， 在 该 选项 卡 中 ， 上 半 部 分 显示 的 是 查询 全 部 数据 的 
SQL 语句 ， 下 半 部 分 是 以 表格 的 形式 显示 的 数据 表 中 的 数据 。 由 于 这 个 数据 表 是 新 创建 的 ， 还 没有 添 
加 任何 数据 ， 所 以 下 面 的 表格 是 空 的 ， 如 图 3.15 所 示 。 


SQL File 1 


国 日 | 多 作 航 吕 I 轩 加 | EP A 


1® IELECT * FROM db_database02.tb_users 


Result Set Filter 个 | Edit [人 2 2 = | ExportImport :四 区 | Wrap Cell Content 五 
i name pwd 


* Ey Cy 





图 3.15 tb_user 选项 卡 


(3) 直接 在 下 面 的 表格 中 添加 需要 的 数据 ， 例 如， 指定 用 户 名 为 mr， 密 码 为 111， 则 可 以 按 如 图 3.16 
所 示 的 样式 进行 添加 。 





3.16 向 数据 表 中 添加 数据 


(4) 单 击 图 3.16 中 的 Apply 按钮 ， 将 弹出 如 图 3.17 所 示 的 对 话 框 ， 显 示 生成 的 可 编辑 的 添加 数 
据 的 SQL 语句 。 
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Review SQL Script iew the SQL Saiptto be Gn 


INSERT INTO “db_database02 .tb_user (“name’, “pwd” ) VALUES (mr, 111'); 


«ii 





3.17 可 编辑 的 添加 数据 的 SQL 语句 
(5) 单 击 Apply 按钮 ， 开 始 添加 数据 。 数 据 添加 完成 后 ， 显 示 如 图 3.18 所 示 的 完成 对 话 框 。 


Applying SQL script to the database ... 


The folowing tasks wil now be executed. Please monitor the exeaution. 
Press Show Logs to see the execution logs. 


Execute SQ Statements 





SQL script was successfully appled to the database. 





图 3.18 添加 数据 完成 对 话 框 
(6) 浏览 添加 成 功 后 的 数据 ， 如 图 3.19 所 示 。 








3.19 ”添加 成 功 后 的 数据 


3.1.4 数据 的 导出 和 导入 
在 MySQL Workbench 的 Local instance MySQL56 选项 卡 中 ， 单 击 左 侧 的 “展开 导航 ”图 标 P， 可 


以 显示 一 些 常用 的 导航 超 链接 ， 其 中 最 上 面 的 MANAGEMENT 栏目 中 ,提供 了 Data Export 和 Data 
Import/Restore 两 个 超 链接 ， 如 图 3.20 所 示 。 


中 
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Navigator 
MANAGEMENT 
【>] Server Status 
星 Client Connedtions 
旦 Users and Privileges 


Bl status and System Variable @ 实现 数据 导出 

点 Data Export 

启 Data Import/Restore @ 实现 数据 导入 
图 3.20 MANAGEMENT 栏目 


通过 这 两 个 超 链接 就 可 以 实现 数据 的 导出 和 导入 功能 了 ， 下 面 分 别 进行 介绍 。 
1. 数据 导出 


导出 数据 库 中 数据 的 具体 步骤 如 下 。 





(1) 在 MySQL Workbench 的 Local instance MySQL56 选项 卡 的 左 侧 的 MANAGEMENT 栏目 中 单 
击 Data Export 超 链 接 ， 将 弹出 输入 root 用 户 密码 的 对 话 框 ， 在 该 对 话 框 中 输入 密码 “root”， 单 击 OK 
按钮 ， 即 可 显示 如 图 3.21 所 示 的 Data Export 选项 卡 。 


是 ms 


Dat Export 


Objecr salecton [Erport ogre 





@ 将 数据 库 中 全 部 数据 表 导 
出 到 指定 的 文件 夹 中 





Lite ] 


ea 





® Eportto ppproctFalde Ge Varnevater Pornens dnps Dea 


Ea tobe wa be erported rt sporate 全 rs ow 2 weectve restore ha may be cower 
Et Sef Cortored re ee 








| 


和 目 将 数据 库 中 选中 的 数据 表 导 出 到 指定 的 SQL 文件 





图 3.21 导出 数据 选项 卡 
(2) 在 导出 数据 选项 卡 中 ， 选 择 要 导出 的 数据 库 ， 如 db_database02; 在 下 面 的 Options 区 域 中 ， 


选中 Export to Self-Contained File 单 选 按钮 ， 并 且 单 击 其 右 侧 的 “.……” 按 钮 ， 设 置 导出 数据 的 存储 位 置 ， 
如 图 3.22 所 示 。 
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目 将 数据 库 中 选中 的 数据 表 导 出 到 db_database02.sql 文件 中 





图 3.22 设置 导出 数据 
(3) 单 击 Start Export 按钮 ， 执 行 导出 操作 ， 导 出 成 功 后 ， 将 显示 如 图 3.23 所 示 的 界面 。 


MysQL 5.6 





Data Export 


[eared Optons..] 
Object Selecton | Export Progress 





Export Completed 





Status: 
1of 1 exported. 
Log: 


09:20:53 Dumping db_database02 (al tables) 

Running: mysqldump.exe --defoultsextra-fle="c: Wsers pdmni~1\ppdata Yocal\temp\impfahbg. cnf” ~user =root -max_alowed._packet=1G -host=127.0.0.1 -port=3307 
default-character-set=utf3 "db_database02" 

09:20:59 Export of D: mysql_data db._database02.sq has fnahed 








tp | [StartExport 


图 3.23 导出 成 功 
2. 数据 导入 


将 数据 库 中 数据 导出 为 SQL 文件 后 ， 还 可 以 把 它 导入 到 MySQL， 具 体 步 又 如 下 。 


(1) 在 MySQL Workbench 的 Local instance MySQL56 选项 卡 的 左 侧 的 MANAGEMENT 栏目 中 ， 
f Data Import/Restore 超 链接 ， 将 弹出 输入 root 用 户 密码 的 对 话 框 ,在 该 对 话 框 中 输入 密码 “root”， 
f OK 按钮 ， 即 可 显示 如 图 3.24 所 示 的 Data Import/Restore 选项 卡 。 
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dminisyaon - Data ImporyRes -xx 











SQL 56 
Data Import 


imoort fiom Dk [import Progress 


@ 指定 的 文件 夹 中 将 数据 库 
的 全 部 数据 表 导 入 





®lmport from Dump Project Folder 
Select the Durp Project Falder to mport. You can do a selecove restore. 
[aase cone 





Ceers dnnst ator Poaments tmps 











| Te TESTEE 


四 通过 指定 的 SQL 文件 导入 


Select Database Objects to Import (orly avalable for Project Folders) 








Imp.., Schema Imp... Schema Objects 


Press [Start Import] to start 





上 稀 开始 导入 按钮 
图 3.24 导入 数据 选项 卡 


(2) 在 导入 数据 选项 卡 中 , 选择 根据 指定 的 SQL 文件 导入 数据 。 例如 , 从 名 称 为 db_database02.sql 
的 SQL 文件 中 导入 数据 。 在 Options 区 域 中 ， 选 中 Import from Self-Contained File 单 选 按钮 ， 并 且 单 击 
其 右 侧 的 “...” 按 钮 ， 选 择 导 入 SQL 文件 的 存储 位 置 ， 如 图 3.25 所 示 。 
种 ‘Data Import 
noort fom ek. [inport Propres 


Optons 
rport hon uro make Foder Ve Weer ty Docenti trp 


加 rport fom set Contned Fie De mysd cata Wotabased2. sd 








Select he SQL /ump fle ip mport Pesse note that the whoke fie wll be rporied, 





Defat scherma obe inoorted To ~ @ 指定 要 导入 数据 的 SQL 文件 
bE TPE Oe Fer PT Fe Gro 
Defouk Target Schema: [Mews | NOTE: th & only used if pe dmp fle doesn't contan its schema, 
ewe te roree, 


Seiect Database Objects to lrport (ory avalaple for Project Folders) 





mp。 Sehema limp... Scene Object 


日 单 击 开始 导入 上 
Press [Start Inport] to stat Start Import 


图 3.25 设置 导入 数据 选项 
(3) 单 击 Start Import 按钮 ， 执 行 导入 操作 。 
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3.2 phpMyAdmin 图形 化 管理 工具 


phpMyAdmin 是 众多 MySQL 图 形 化 管理 工具 中 应 用 最 广泛 的 一 种 ， 是 一 款 使 用 PHP 开发 的 B/S 
模式 的 MySQL 客户 端 软件 ， 该 工具 是 基于 Web 跨 平台 的 管理 程序 ， 并 且 支 持 简体 中 文 。 用 户 可 以 在 
官方 网 站 www.phpmyadmin net 上 免费 下 载 到 最 新 的 版 本 ， 本 书 中 使 用 最 新 版 本 phpMyAdmin 4.2.8。 
phpMyAdmin 为 Web 开发 人 员 提供 了 类 似 于 Access、SQL Server 的 图 形 化 数据 库 操 作 界 面 ， 通 过 该 管 
理工 具 可 以 完全 对 MySQL 进行 操作 ， 例 如 ， 创 建 数据 库 、 数 据 表 、 生 成 MySQL 数据 库 脚 本 文件 等 。 

在 浏览 器 地 址 栏 中 输入 http:/WlocalhostphpMyAdmin/， 在 弹出 的 对 话 框 中 输入 用 户 名 和 密码 ， 进 入 
phpMyAdmin 图 形 化 管理 主 界面 , 接 下 来 就 可 以 进行 MySQL 数据 库 的 操作 。 下 面 将 分 别 介绍 如 何 创 建 、 
er 
0 培 明 

应 用 phpMyAdmin 图 形 化 管理 工具 有 一 个 前 提 条 件 ， 必 须 在 本 机 中 搭建 PHP 运行 环境 ， 将 其 
作为 一 个 项 目 在 PHP 开发 环境 中 运行 应 用 。 如 果 通 过 PHP 的 集成 化 安装 包 来 搭建 PHP 运行 环境 ， 
那么 在 这 个 安装 包 中 就 已 经 包含 phpMyAdmin 图 形 化 管理 工具 ， 环 境 搭建 完成 后 即 可 直接 使 用 。 


3.2.1 配置 phpMyAdmin 


1. 解压 文件 到 指定 目录 


将 下 载 好 的 phppMyAdmin 解压 缩 并 复制 到 Web 服务 器 的 文档 根 目录 下 。 例 如 ， 使 用 Apache 作为 
Web 服务 器 ，Apache 的 文档 根 目录 为 F:\wamp\webpage， 则 将 解压 好 的 phpMyAdmin 文件 夹 复 制 到 
F:\wamp\webpage 即 可 ， 如 图 3.26 所 示 。 
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图 3.26 复制 到 文档 根 目录 
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2. 创建 config.inc.php 文件 


在 phpMyAdmin 文件 夹 中 找到 config.sample.inc.php 文件 ， 将 其 改名 为 config inc.php。 
在 浏览 器 地 址 栏 中 输入 http:WlocalhosVphpMyAdmin， 如 图 3.27 所 示 ， 输 入 数据 库 的 用 户 名 密码 
登录 。 





[€ 加 [Erp//ocadhos/pt D - CX 四 da 














phpMUyAdrmiIn 
欢迎 使 用 phpMyAdmin 


rg 


中 文 -Chnese simplised 。。 国 | 





[se) 


用 户 名 : 

















图 3.27 登录 phpMyAdmin 
登录 成 功 可 看 到 如 图 3.28 所 示 界 面 。 
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图 3.28 成 功 登 录 phpMyAdmin 界面 


第 3 章 使 用 MySQL 图 形 化 管理 工具 





登录 后 看 到 有 如 图 3.29 所 示 的 字样 “phpMyAdmin 高 级 功能 尚未 完全 设置 ， 部 分 功能 未 激活 ， 请 
这 里 查看 原因 ”， 则 执行 如 下 步骤 。 


| 和 phpMyAdmin 高 级 功能 尚未 完全 设置 ， 部 分 功能 未 激活 。 请 点 击 这 里 查看 原因 














图 3.29 phpMyadmin 高 级 功能 尚未 完全 设置 提示 


(1) 单 击 “ 导 入 ”标签 ， 然 后 单 击 “ 浏 览 ” 按 钮 ， 找 到 phpMyAdmin/examples/create_table.sql 文 
件 ， 将 它 导 入 ， 如 图 3.30 和 图 3.31 所 示 。 


DIE 
) 问世 库 月 sal 副 其 夺 “ 用 户 妃 SL 园 SA 8 于 页 


导入 到 当前 服务 器 


要 时 入 的 文件 
i 

从 计算 性 巾 上 传 。 CIUSSSRRNDESNGRCR C2] (电大 限 抽 :8.132 KB) 
六 位 的 字 答 委 。UE = 


部 分 导入 : 
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从 篆 一 个 开 技 际 过 的 查询 效 (SQL 同 ) 或 行 类 1 二 他 用 ) 0 


格式 :> 
SGL 





图 3.30 导入 create table.sql 文件 


数据 床 回 SQL 克 状 丰 =: 用 户 局 S 出 局 SA 六 设 和 上 串 圳 口交 里 硬 闻 特集 兮 引 
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图 3.31 导入 create table.sql 文件 成 功 
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(2) 将 config.inc.php 文件 中 的 以 下 内 容 的 注释 去 掉 。 


$cfg['Servers][$ilrpmadb] = phpmyadmin '; 
$cfg['Servers][$ilrbookmarktable] = 'pma__bookmark':; 
$cfg['Servers][$il[relation] = 'pma__relation'; 
$cfg['Servers][$il[table_info] = 'pma__table_info' 
$cfg['Servers'][$il[table_coords'] = 'pma_table_coords'; 
$cfg[Servers][$il[pdf_pages] = 'pma_pdf_pages'; 
$cfg['Servers][$ilrcolumn_info] = 'pma__column_info’; 
$cfg['Servers][$ilrhistory] = 'pma_history'; 
$cfg['Servers][$il[table_uiprefs] = 'pma__table_uiprefs'; 
$cfg['Servers'][$il[tracking] = 'pma_tracking'; 
$cfg['Servers'[$il['designer_coords'] = ‘pma_designer_coords'; 
$cfg['Servers][$ilruserconfig] = ‘pma__userconfig'; 
S$cfg['Servers'][$il[recent] = ‘pma__recent’; 
$cfg['Servers'][$il['favorite] = 'pma__favorite'; 
S$cfg['Servers'][$il['users'] = 'pma_users' 
$cfg['Servers'][$i]['usergroups'] = 'pma_usergroups' 
S$cfg['Servers'][$i]['navigationhiding] = 'pma_navigationhiding '; 
$cfg['Servers'"][$il['savedsearches'] = 'pma__savedsearches'; 


(3) 修改 phpMyAdmin/libraries/config.default.php 文件 ， 内 容 如 下 。 


$cfg['Servers][$il['controluser] = "username'; ”/* 数 据 库 用 户 名 */ 
$cfg['Servers'][$il['controlpass] = 'password'; ”/* 数 据 库 密码 */ 
$cfg['Servers][$ilrpmadb] = 'phpmyadmin'; 
$cfg['Servers'][$ilrbookmarktable] = 'pma_bookmark'; 
$cfg['Servers'][$il[relation] = 'pma_relation '; 
$cfg['Servers'][$il[table_info] = 'pma_table_info'; 
$cfg['Servers'][$il[table_coords'] = 'pma_table_coords'; 
$cfg['Servers][$ilrpdf_pages] = 'pma_pdf_pages'; 
$cfg['Servers'][$il['column_info] = 'pma_column_info'; 
$cfg['Servers'][$il[history] = ‘pma_history’; 
$cfg['Servers][$il[designer_coords] = ‘pma_designer_coords'; 
S$cfg['Servers'][S$il['recent] = ‘pma_recent'; 
$cfg['Servers][$il[table_uiprefs] = pma_table_uiprefs'; 
$cfg['Servers'][$il[tracking] = 'pma_tracking '; 
$cfg['Servers'][$il[ruserconfig] = 'pma_userconfig'; 


3.2.2 数据库 操 作 管理 


1. 创建 数据 库 


在 phpMyAdmin 的 主 界面 ， 首 先 在 文本 框 中 输入 数据 库 的 名 称 db_study， 然 后 在 下 拉 列 表 框 中 选 











择 所 要 使 用 的 编码 ， 一 般 选 择 gb2312_chinese_ci 简体 中 文 编码 ， 单 
图 3.32 所 示 。 成 功 创建 数据 库 后 ， 将 显示 如 图 3.33 所 示 的 界面 。 











ff “创建 ”按钮 ， 创 建 数据 库 ， 如 
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数据 库 


@ 单 击 “创建” 按钮 ， 创 建 数 据 库 | 














@ 选择 数据 库 排序 规则 








名 新 建 数 据 库 豆 
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information_schema 

mysql 

performance_schema 
器 phpmyadmin 

总 计 : 4 


utf8_general_ci 检查 权限 
1atin1l_swedish_ci ai 检查 权限 
utf8_general_ci | 检查 权限 
utf8_bin ai 检查 权限 
latinl_swedish_ci 


项 国 刚 际 














phpMyAdmin 


全 加 也 避 站 全 
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三 对 Now 


由 pnpmradmm 








成 功 创建 数据 库 db_study 











scan 


3.33 数据库 创 建成 功 


| 在 右 合 界面 中 可 以 对 该 数据 库 进 行 相关 操作 ， 如 结构 、SQL、 搜 索 、 查 询 、 导 出 、 导 入 、 权 限 
， 等， 单 击 相应 的 标签 进入 相应 的 操作 界面 。 
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2. 修改 、 删 除数 据 库 
在 如 图 3.34 所 示 的 界面 中 ， 在 右 侧 界面 还 可 以 对 当前 数据 库 进行 修改 。 单 击 界面 中 的 “操作 ” 标 
签 ， 进 入 修改 操作 页 面 。 












| hfiocalhos Ehprmye: 


phpMyAdmin 


_| @ 单 击 该 超 链接 即 可 
-| 删除 数据 库 














@ 可 以 向 数据 库 db_study 添加 新 表 ， 也 可 以 对 
数据 库 重 命名 ， 单 击 相 应 的 “执行 ”按钮 即 可 























图 3.34 ”修改 数据 库 


(1) 可 以 对 当前 数据 库 执行 创建 数据 表 的 操作 ， 只 要 在 创建 数据 表 的 提示 信息 下 面 的 两 个 文本 
框 中 分 别 输入 要 创建 的 数据 表 的 名 称 和 字段 总 数 ， 然 后 单 击 “ 执 行 ”按钮 即 可 进入 到 创建 数据 表 结 
构 页 面 。 

(2) 也 可 以 对 当前 的 数据 库 重 命名 ， 在 Rename database to: 下 的 文本 框 中 输入 新 的 数据 库 名 称 ， 
单 击 “ 执 行 ”按钮 ， 即 可 成 功 修改 数据 库 名 称 。 

(3) 也 可 以 单 击 “ 删 除数 据 库 ” 超 链接 来 删除 该 数据 库 。 








3.2.3 ”管理 数据 表 





管理 数据 表 是 以 选择 指定 的 数据 库 为 前 题 ， 然 后 在 该 数据 库 中 创建 并 管理 数据 表 。 下 面 就 来 介绍 
如 何 创建 、 修 改 和 删除 数据 表 。 

1. 创建 数据 表 

创建 数据 库 db_study 后 ， 在 右 侧 的 操作 页 面 中 输入 数据 表 的 名 称 和 字段 数 ， 然 后 单 击 “ 执 行 ” 按 
钮 ， 即 可 创建 数据 表 ， 如 图 3.35 所 示 。 
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图 3.35 创建 数据 表 


成 功 创建 数据 表 tb_admin 后 , 将 显示 数据 表 结 构 界面 。 在 表单 中 对 各 个 字段 的 详细 信息 进行 录入 ， 
包括 字段 名 、 数 据 类 型 、 长 度 / 值 、 是 否 为 空 、 主 键 、 是 否 自 增 等 ， 以 完成 对 表 结 构 的 详细 设置 。 当 所 
有 的 信息 都 输入 以 后 ， 单 击 “ 保 存 ” 按 钮 ， 创 建 数据 表 结 构 ， 如 图 3.36 所 示 。 成 功 创建 数据 表 结 构 后 ， 
将 显示 如 图 3.37 所 示 的 界面 。 
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A 成 功 创建 数据 库 表 结构 
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图 3.37 成 功 创建 数据 表 
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2. 修改 数据 表 


一 个 新 的 数据 表 被 创建 后 ， 进 入 到 数据 表 页 面 中 ， 在 这 里 可 以 通过 改变 表 的 结构 来 修改 表 ， 可 以 
执行 添加 新 的 列 、 删 除 列 、 索 引 列 、 修 改 列 的 数据 类 型 或 者 字段 的 长 度 / 值 等 操作 ， 如 图 3.38 所 示 。 








|- 于 | 修改 字段 属性 
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选中 各 字段 前 的 复 选 框 ， 批 量 编辑 数据 表 结构 





图 3.38 ”修改 数据 表 结构 


3. 删除 数据 表 


要 删除 某 个 数据 表 ， 单 击 页 面 中 的 “操作 ”标签 ， 之 后 单 击 页 面 中 红色 的 超 链接 “删除 数据 表 ” 
即 可 成 功 删 除 指定 的 数据 表 ， 如 图 3.39 所 示 。 
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3.39 ”删除 数据 表 结 构 


3.2.4 ”管理 数据 记录 








击 phpMyAdmin 主 界面 中 的 SQL 标签 , 打开 SQL 语句 编辑 区 。 在 编辑 区 输入 完整 的 SQL 语句 ， 
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来 实现 数据 的 查询 、 插 入 、 修 改 和 删除 等 操作 。 


1. 使 用 SQL 语句 插入 数据 
在 SQL 语句 编辑 区 使 用 INSERT 语句 向 数据 表 tb_admin 中 插入 数据 后 ， 单 击 “ 执 行 ”按钮 ， 向 数 
据 表 中 插入 一 条 数据 ， 如 图 3.40 所 示 。 如 果 提 交 的 SQL 语句 有 错误 ， 系 统 会 给 出 一 个 警告 ,提示 用 户 
修改 它 ， 如 果 提 交 的 SQL 语句 正确 ， 则 弹出 如 图 3.41 所 示 的 提示 信息 。 
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在 数据 库 db_study 运行 SQL 查询 : 全 @ 单 击 SQL 标签 


pud")valuea ("mr', "mrsoft" 









insert into cb_admin (name' 








@ 使 用 insert 语句 向 数据 表 tb_admin 


中 插入 数据 , 可 以 应 用 右 侧 列表 来 选 
择 要 操作 的 列 





SELECT") [SELECT) (INSERT) LUPDATE ) (“DELETE ) (清除 


Bookmark this SQL query 
全 单 击 “ 执 行 ”按钮 添加 数据 到 数据 表 加 


【语句 定 界 符 ] 问 在 此 再 次 旺 示 此 查 词 保留 查询 杠 

















图 3.40 使 用 SQL 语句 向 数据 表 中 插入 数据 


4 本 一 ~ id name pwd 
管 至 到 ID 号 。。 管理 员 名 短 管理 两 宫 王 
回 sxP 编辑 3 复制 @ 删除 1 mr mrsoft 


图 3.41 成 功 添加 数据 信息 


/ 
说 明 
为 了 编写 方便 ， 可 以 利用 其 右 侧 的 属性 列表 来 选择 要 操作 的 列 ， 只 要 选中 要 添加 的 列 ， 双 击 其 
选项 或 者 单 击 << 按 钮 添加 列 名 称 。 
2. 使 用 SQL 语句 修改 数据 


在 SQL 语句 编辑 区 应 用 UPDATE 语句 修改 数据 信息 ,将 ID 为 1 的 管理 员 的 名 称 改 为 “明日 科技 ”， 
密码 改 为 111， 添 加 的 SQL 语句 如 图 3.42 所 示 。 
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浏览 泣 续 构 司 SQL 邯 执 索 张 括 A 时 Ss 同 导 入 = 权限 咏 季 作 全 这 中 三 前 发 器 


在 数据 诛 db_study 运行 SQL 但 丙 : 司 





修改 数据 表 信息 


SELECT*) (SELECT) INSERT】 “UPDATE) [DELETE ) (清除 


Bookmark ms SQL query 


[语句 定 界 答 ] 回 在 此 再 次 显示 此 查询 门 保留 查 询 各 





图 3.42 修改 数据 信息 的 SQL 语句 
单 击 “ 执 行 ”按钮 ， 数 据 修 改 成 功 。 比 较 修改 前 后 的 数据 如 图 3.43 所 示 。 


修改 id=1 
的 数据 


id name 
Yasos Mle Se 





3.43 ”修改 单条 数据 的 实现 过 程 
3. 使 用 SQL 语句 查询 数据 


在 SQL 语句 编辑 区 应 用 SELECT 语句 检索 指定 条 件 的 数据 信息 ， 将 DD 小 于 4 的 管理 员 全 





出 来 ， 添 加 的 SQL 语句 如 图 3.44 所 示 。 


] 济 0 结构 SQL 不 搜索 驶 后 A 屋 Sg 司 呈 入 民用 产 报 作 于 这 后 区 地 发 台 





在 数据 诛 db_stuuy 运行 SQL 直 调 如 





查询 数据 表 信息 


SELECT (SELECT) (NSERT) (UPDATE) (DELETE) (天 这 


Baokmare this SDL query 


【证 司 定 界 和 ] 风 在 此 再 次 旺 示 此 音调 站 保 窗 可 将 


111 


执行 


执行 








图 3.44 查询 数据 信息 的 SQL 语句 


S2 





部 


显示 
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单 击 “执行 ”按钮 ， 该 语句 的 实现 过 程 如 图 3.45 所 示 。 




















PT vf wd 

aaos |s3sse Ease i a tt 
加 v 编 辑 了 复制 @ 列队 1 图 B 科 技 “111 ma ee 
器 o? 编 辑 对 夏 制 人 @ 出 际 2| 天 得 woh 的 数据 信 加 oF 编辑 了 复制 全 王 除 1 明日 科技 111 
回 编辑 了 夏 制 加 出 际 3 上 痉 鸡 xhy 息 、 门 少 玉 可 苇 丰 制 全 删除 2 无 语 a 
口 少 编 辑 对 夏 制 加 出 除 ai wk 回 上 编辑 了 路 制 @ 出 除 3 轻 焉 xhy 
回 忆 编 辑 对: 复制 @ 列队 5| 匀 六 下 zx 

查询 前 查询 后 





























图 3.45 查询 指定 条 件 的 数据 信息 的 实现 过 程 


除了 对 整个 表 的 简单 查询 外 , 还 可 以 执行 复杂 的 条 件 查询 (使 用 WHERE 子 句 提交 LIKE、 ORDER 
BY、GROUP BY 等 条 件 查询 语句 ) 及 多 表 查 询 ， 读 者 可 通过 实践 灵活 运用 SQL 语句 功能 。 


4. 使 用 SQL 语句 删除 数据 


在 SQL 语句 编辑 区 应 用 delete 语句 检索 指定 条 件 的 数据 或 全 部 数据 信息 ， 删 除名 称 为 “小 科 ” 的 
管理 员 信息 ， 添 加 的 SQL 语句 如 图 3.46 所 示 。 





删除 数据 表 指定 信息 


SELECT (SELECT USERD (UPOATE) ‘DELETE Hi 


oeman ea SQL ae 


1 1 NR 和 本 





3.46 删除 指定 数据 信息 的 SQL 语句 


$6 注 意 
如 果 DELETE 语句 后 面 没 有 WHERE 条 件 值 ， 那 么 将 删除 指定 数据 表 中 的 全 部 数据 。 





单 击 “ 执 行 ”按钮 ， 弹 出 确认 删除 操作 对 话 框 ， 单 击 “确定 ”按钮 ， 执行 数据 表 中 指定 条 件 的 删 
除 操作 。 该 语句 的 实现 过 程 如 图 3.47 所 示 。 

















TT— 如 可 name pwd 
0 ,T， i I 

日 少 编辑 眉 夏 制 @ 出 了 2| 径 再 xhy nane=" 小 saa0s sae Eases 

口 。7 坑 组 对 : 夏 制 @ 出 除 3 无 语 wh pi 固 少 编辑 苇 夏 制 @ 出 除 2 轻 丽 xhy 
[EB 2 © 恒 际 4 小 科 ww 三 一旦 编辑 卫 夏 伸 合 删除 3 无 滞 wgh 

> = 目 。 编 各 了 夏 钊 人 @ 出 除 5 转注 夏 。 zx 
口上 机 大 人 加 制订 5 页 zx 日 3 
管理 员 信息 表 数 据 删除 数据 后 的 结果 
































图 3.47 删除 指定 条 件 的 数据 信息 的 实现 过 程 
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5. 通过 form 表单 插入 数据 


选择 某 个 数据 表 后 ， 单 击 “插入 ”标签 ， 进 入 插入 数据 界面 ， 如 图 3.48 所 示 。 在 界面 中 输入 各 字 
段 值 ， 单 击 “执行 ”按钮 即 可 插入 记录 。 默 认 情 况 下 ， 一 次 可 以 插入 两 条 记录 。 








phpMyAdmin 
l 会 司 3 
近期 访问 未 收 藏 夫 役 关于 本 - 上 添加 各 字段 值 ， 


其 中 id 是 自动 编 
号 ， 可 以 填写 ， 也 
可 以 不 写 


pwd varchar(50) 国 wxk 
@ 单 击 “ 执 行 ”按钮 即 可 插入 记录 


图 3.48 插入 数据 








6. 浏览 数据 


选择 某 个 数据 表 后 ， 单 击 “ 浏 览 ” 标签 ,进入 浏览 界面 ,如 图 3.49 所 示 。 单 击 每 行 记录 中 的 编辑 
超 链接 ， 可 以 对 该 记录 进行 编辑 ， 单 击 每 行 记录 中 的 旧 m 味 超 链接 ， 可 以 删除 该 条 记录 。 

















i tp focalhosyphpmyad P ~ OX 








| 二 econy localhost / db x 










phpMyAdmin [区 服务 器 localhos ~ 全 二 哲 db_study » 饥 到 由 admin 
全 到 B90 (Cm Bs a EAD 
近期 访问 、 玫 惧 峰 夫 
Now EE 在 明示 第 0-3 行 ( 共 4 行 理光 K 可 00000 协 -) 
Be sy | 


SETPCT * FROM ,rn_nanin- 
性 可 分 析 [ 问 二 电 温 ][ 员 辜 [第 新 SOL JL 他 污 PHP 八 码 ] [8 而] 


行 教 : “25 回 过 站 和 
@ 单 击 “ 编 辑 ” 超 链接 


编辑 该 条 记录 
@ 单 击 “ 复 制 ” 超 链接 | 
复制 该 条 记录 















@ 选 中 记录 前 的 复 | 
选 框 , 对 选择 的 多 条 | 吾 @ 单 击 “ 删 除 ” 超 链接 
记录 同时 进行 修改 | + 0D=s =rm [Dm om 删除 该 条 记录 
或 删除 操作 
















图 3.49 浏览 数据 


7. 搜索 数据 
选择 某 个 数据 表 后 ， 单 击 “ 搜 索 ” 标 签 超级 链接 ， 进 入 搜索 页 面 ， 如 图 3.50 所 示 。 在 这 个 页 面 中 ， 
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可 以 在 选择 字段 的 列表 框 中 选择 一 个 或 多 个 列 ， 如 果 要 选择 多 个 列 ， 先 按 下 Ctrl 键 并 单 击 要 选择 的 字 
段 名 ， 查 询 结果 将 按照 选择 的 字段 名 进行 输出 。 

在 该 界面 中 可 以 对 记录 按 条 件 进 行 查询 。 查 询 方式 有 两 种 : 第 一 种 方式 使 用 依 例 查询 。 选 择 要 查 
询 的 条 件 ， 并 在 文本 框 中 输入 要 查询 的 值 ， 单 击 “执行 ”按钮 。 第 二 种 方式 选择 构建 WHERE 语句 查 
询 。 直 接 在 “where 语句 的 主体 ”文本 框 中 输入 查询 语句 ， 然 后 单 击 其 后 的 “执行 ”按钮 。 



















CA nepyvlocalhosyphpmyadr 训 -Ex | 娩 Iocalhost / localhost / db_x 


ph 

















pMyAdmin 一 SL 
全 加 局 昌 站 @ ] 浏览 Wm so [ ~ sr | EX 及 5 局 导入 =5 权限 vv 更 多 
近期 访问 表 收 藏 
B New 字段 类 型 。 排序 规则 运算 符 什 
四 -3 db_study id int(4) 和 
LB 亲 建 
tL tb_admin name i gbk_chinese_ci LIKE 
(50) 









由 -3 information_schema 
由 mysql 

由 peromance_schema 
由 phpmyadmin 


@ 方 式 一 : 使 用 依 例 查询 。 选 
择 要 查询 的 条 件 和 输入 要 查 
询 的 值 ， 单 击 “ 执 行 ”按钮 


四 选择 查询 结果 显 
示 的 字段 名 称 






pwd varchar gbk_chinese ci (LIKE = 
(50) 加 











加 






选择 字段 《至少 一 个 ) : 残 膛 加 搜索 条 件 〈“where" 从 句 的 主体 ) : 


图 : 


ee @ 方 式 二 : 使 用 WHERE 子 句 。 在 









文本 框 中 输入 查询 条 件 ， 单 击 “ 执 
行 ”按钮 





执行 











图 3.50 ”搜索 查询 


3.2.5 ”导出 导入 数据 


导出 和 导入 MySQL 数据 库 脚本 是 互 逆 的 两 个 操作 。 导 出 是 将 数据 表 结 构 、 表 记录 存储 为 .sql 的 脚 
本 文件 ， 导 入 是 执行 扩展 名 为 “.sql” 的 文件 ， 将 数据 导入 到 数据 库 中 。 通 过 导入 和 导出 的 操作 实现 数 
据 库 的 备份 和 还 原 。 


MySQL 从 入 门 到 精通 


1. 导出 MySQL 数据 库 脚本 

单 击 phpMyAdmin 主 界面 中 的 “导出 ”标签 ， 打 开导 出 编辑 区 ， 如 图 3.51 所 示 。 选 择 导 出 文件 的 
格式 ， 这 里 默认 使 用 选项 SQL,， 单 击 “ 执 行 ” 按钮， 弹出 如 图 3.52 所 示 的 “另存 为 ”对 话 框 ， 单 击 “ 保 
存 ” 按 钮 ， 将 脚本 文件 以 “.sql” 格 式 存储 在 指定 位 置 。 



































[i hep/ocalhos pnpmsadr P ~ © X | ji lecahosyiocalhes/db-x ee 
phpMyAdmin 国 | [而 务 靶 : )ocalhost 和 歼 提 库 -db_study 四 
全副 双生 站 @ 3 结构 当 SQ 人 拓 过 再 7 呈 山本 本 @ 厅 了 黑 
过 央 访 可 才 jk 攻 天 
WE 3 a 站 证 岂 9 
ee 正在 导出 数据 库 *db_study" 中 的 数据 表 
六 党 
-7 Wb_aomin 导出 方式 : 
PB_ 4 informaton_schema 
ms 人 快 带 -显示 最 4 的 过 而 引 | 
performance_schema 自 定义 -时 示 所 有 可 用 8 大 
由 phpmyadmin 




















人 @ 单 击 “ 执 行 ” 按 
(ee | 钮 ， 即 可 生成 脚本 ly 
图 3.51 生成 MySQL 脚本 文件 设置 界面 
CT 
生 口 -四 是 ， || 这 ss 7 











组 织 ” 。 新 建文 件 夫 村 " @ 

4 广 收 训 天 “Internet 快捷 方式 (1) 5 
如 下 载 Internet Explorer 疑难 解答 Ll 
FT 川 詹 | Internet 快 于 方式 上 
1010 L 134 字 方 a 
辐 最 访问 的 位 置 | “PostgresQL (1) 


人 create tables.sql 


Ee PostgreSQL 
库 
转 9.47 KB 





4 曾 计算 机 “4 快捷 方式 (1) 
bp 总 本 地 开盘 (C) ee - 快 于 方式 
? 本 地 开盘 (Dj 辆 | i 
?加 本 地 三 盘 (E) ”~ 一心- vv hd 
文件 名 (N): db_study.sql 和 
保存 类型 (nD: [PostgreSQL (sq | 














二 了 六 文人 闪 有 


3.52 ”另存 为 MySQL 脚本 对 话 框 














第 3 章 使 用 MySQL 图形 化 管理 工具 


2. 导入 MySQL 数据 库 脚本 


单 击 “ 导 入 ”标签 ， 进 入 执行 MySQL 数据 库 脚 本 界面 ， 单 击 “ 浏 览 ” 按 钮 查找 脚本 文件 〈 如 
db_ study.sql) 所 在 位 置 ， 如 图 3.53 所 示 ， 单 击 “ 执 行 ” 按 钮 ， 即 可 执行 MySQL 数据 库 脚本 文件 。 














WW 络 鬼 自 sSa 习 拓 家 可 当局 SU SA 要 志和 加 囊 相 i 7 


@ 单 击 “ 导 入 ”标签 


导入 到 数据 库 “db_study” 





要 导入 的 文件 : 


文件 可 能 已 压缩 (gzip. zip) 或 未 压缩 。 
压 编 文件 名 必须 以 , 略 式 ].[ 压 综 方 式 ] 结尾 。 如 : .sqLzip 


从 计算 机 中 上 传 : ”CAUsersipkhiDesktopWdb (最 放 RR 制 : 3.192 KB) 








二 @ 选 择 MySQL 数据 库 脚本 文件 








Ts 回 ] 


部 分 导入 : 
回 在 导入 时 周 本 若 检测 氏 | 可 能 需要 花 珊 很 长 时 间 〈 按 近 PHPiB8d63R 定 ) 则 允许 中 断 * ( 尽 亲 过 会 六 半 坟 分 ， 合 记 曙 人 天文 他 结 旦 个 偶 针 对 方 学。 ) 
从 第 一 个 开 焙 跳 ja9 查 调 数 ( SQL 用 ) 或 行 数 (其 他 用 ) : |0 


格式 : 
SQL 回 

四 选择 SQL 类 型 
格式 特定 渤 项 : 


SQL 兼 百 模 式 : | NONE 回 
园 不 要 纵 零 值 使 用 自 增 (ADTO 


(G5 | 全 单 击 “ 执 行 ”按钮 即 可 生效 


3.53 执行 MySQL 数据 库 脚本 文件 








人 注意 
在 执行 MySQL 脚本 文件 前 ， 首 先 检测 是 否 有 与 所 导入 数据 库 同名 的 数据 库 ， 如 果 没有 同名 的 
数据 库 ， 则 首先 要 在 数据 库 中 创建 一 个 名 称 与 数据 文件 中 的 数据 库 名 相同 的 数据 库 ， 然 后 再 执行 
MySQL 数据 库 脚本 文件 。 另 外 ， 在 当前 数据 库 中 ， 不 能 有 与 将 要 导入 数据 库 中 的 数据 表 重 名 的 数 
据 表 存 在 ， 如 果 有 重 名 的 表 存 在 导入 文件 就 会 失败 ， 提 示 错 误 信息 。 | 


Nn 


读者 可 也 可 通过 单 击 phpMyAdmin 图 形 化 工具 左 侧 区 的 g 按 钮 ,在 打开 的 对 话 框 中 , 单 击 “ 导 
， 入 文件 ” 超 链接 ， 然 后 选择 脚本 文件 所 在 的 位 置 ， 从 而 执行 脚本 文件 。 
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3.2.6 ” phpMyAdmin 设置 编码 格式 


将 页 面 、 程 序 文件 、 数 据 库 与 数据 表 设 置 统一 的 编码 格式 可 以 使 程序 运行 时 不 至 于 出 现 乱码 。 一 
般 情况 下 ， 设 置 页 面 的 编码 格式 由 HIML 中 的 meta 标签 实现 ， 设 置 程序 文件 的 编码 格式 是 由 headerO 
函数 实现 ， 设 置 数据 库 与 数据 表 的 编码 格式 可 以 通过 使 用 phpMyAdmin 实现 。 下 面 以 实例 详细 讲解 一 
下 如 何 为 新 创建 的 数据 库 设 置 编 码 格式 。 具 体 步 又 如 下 。 
(1) 登录 到 phpMyAdmin 图 形 化 工具 页 面 , 创建 数据 库 名 称 , 并 为 新 创建 的 数据 库 选 择 编码 格式 ， 
如 图 3.54 所 示 。 


| http://ocalhost/pt P - © X | localhost /localhost | ph.. x | 





















局 推荐 使 用 广告 过 小 功能 , 去 除 垃圾 广告 , 打开 网 页 更 快 更 干净 ! 马上 使 用 |。 不 再 提示 x 
phpMyAdmin a 
会 一 忆 而 自 @ i 数据库 与 SQL 局 拓 态 = 用 户 区 SH 加 从 证 7 可 
近期 访问 表 收 藏 夫 
Bd New a 
3 db_study 数据 库 
Fim 
由 tb_admin 
电 -号 nbmaton_zchema 
由 .mysql 
wtfe_danish_ci 
申 Ee 让 注意 : 在 比 语 娄 据说 v3-esperanto_ cl 睛 器 之 9 帝国 吾 增 。 
由 -ji phpmyadmin wtf3_general_ci 
Ut J 500. 
数据 库 。 排 H ee ci 
四 db_study gb: 于 cl 
US_celandk_cl 
information_schema "wtf8_latvian_ci 
wt-_ Mhuanian_ci 
mysql la utfe_persian_ci 
perommance schema = ee 
日 wpmaamn a 
总 计 :5 到 ue-sovak @@ 通 过 下 拉 列 表 框 为 新 
1 日 全 阅 有 | 创建 的 数据 库 设 置 编 
. i Ut-Swed 
启用 统计 Ma 码 格式 
[ET 
Ut8mb4_bi 
USmD4 5 cl 
utfemb4_czech_ci 
temb4_danish ci bd 
































图 3.54 设置 数据 库 的 编码 格式 
(2) 创建 数据 表 ， 定 义 数据 表 字段 ， 并 为 新 创建 的 数据 表 设 置 编码 格式 ， 如 图 3.55 所 示 。 


人 
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图 3.55 设置 字段 编码 格式 


3.2.7 “phpMyAdmin 添加 服务 器 新 用 户 


在 phpMyAdmin 图 形 化 管理 工具 中 ， 不 但 可 以 对 MySQL 数据 库 进行 各 种 操作 ， 而 且 可 以 添加 服 
务 器 的 新 用 户 ， 并 对 新 添加 的 用 户 设 置 权 限 。 

在 phpMyAdmin 中 添加 MySQL 服务 器 新 用 户 的 步 又 如 下 。 

(1) 单 击 phpMyAdmin 主 界面 中 的 “用 户 ” 标 签 ， 打 开 服务 器 用 户 操作 界面 ， 如 图 3.56 所 示 。 


ei 





BAI, ) 
MI 











图 3.56 服务 器 用 户 一 览 表 
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mr 











(2) 在 该 界面 中 ， 单 击 “ 添 加 用 户 ” 超 链接 。 进 入 到 如 图 3.57 所 示 界 面 ， 设 置 用 户 名 、 密 码 、3 
机 ， 并 对 新 用 户 的 权限 进行 设置 。 设 置 完成 后 ， 单 击 “ 执 行 ” 按 钮 ， 完 成 对 新 用 户 的 添加 操作 ， 返 回 


互 








主页 面 ， 将 提示 新 用 户 添加 成 功 。 


= 名 、 主 机 、 密 码 








执行 新 用 户 添加 








图 3.57 设置 添加 用 户 信息 


3.2.8 ” phpMyAdmin 中 重 置 MySQL 服务 器 登录 密码 


在 phpMyAdmin 图 形 化 管理 工具 中 ， 不 但 可 以 对 MySQL 数据 库 进行 各 种 操作 ， 而 且 可 以 对 用 户 
的 权限 进行 设置 ， 同 时 还 可 以 对 MySQL 服务 器 的 登录 密码 进行 重 置 。 

在 phpMyAdmin 中 重 置 MySQL 服务 器 登录 密码 的 步骤 如 下 。 

(1) 单 击 服务 器 用 户 操作 界面 ， 如 图 3.58 所 示 。 














EL 


A 共用 户 ，) 
A 














图 3.58 服务 器 用 户 一 览 表 
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(2) 在 该 界面 中 ， 可 以 对 指定 用 户 的 权限 进行 编辑 、 可 以 添加 新 用 户 和 删除 指定 的 用 户 。 这 里 选 
择 指 定 的 用 户 ， 单 击 命 编 x 限 超 链接 ， 对 指定 用 户 的 权限 进行 设置 ， 进 入 到 如 图 3.59 所 示 界 面 。 


Se A htpy/localhostipt P = © | Ao localhost/ localhost 1p.. * | Ow 
中 服务 器 : localhost 加 “| 
































子 数据 库 已 SQL 县 失 本 用户 已 SH 电信 A 8i Y 到 多 
PER User groups 
用 户 概况 
用 户 主机 密码 全 RR 9 User group 授权 
园 任意 % — USAGE 否 局 闹 加 权限 局 SU 
口 任意 localhost 否 UsAGE 否 已 闹 有 阴 己 届 
加 root 127.001 否 。 MIL BRIVILEGES 是 局 疙 强权 限 图 St 翌 
口 rat 了 否 。 ALL SRIVILEGES 是 “已 病 有 限 已 St 
回 root localhost 是 ALL FRIVILEGES 是 切 洗 掉 p 限 加 号 此 
加 % 是 DsAGE 否 ”总 着 包公 限 吕 号 出 
人 回 全 迁 。 才 +#* 严 国 叶 出 
如 添加 用 户 
| MH | 
伍 销 用 户 所 有 祝 限 ， 然 后 条 用 户 。) 
日 星 和 与 用 户 同名 的 娄 掺 际 。 
执行 

















图 3.59 编辑 用 户 权 限 
单 击 “修改 密码 ”按钮 ， 进 入 到 如 图 3.60 所 示 界 面 。 











Edit Privileges: 用 户 "@'ocalhost 


= 
es mA:| | 


BN 方式: MySQL 41+ 
MySQL40 兼 容 


生成 密码 生成 


| 

















图 3.60 修改 密码 
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在 如 图 3.60 所 示 的 界面 中 ， 可 以 设置 用 户 的 权限 、 修 改 密码 、 更 改 登录 用 户 信息 和 复制 用 户 。 在 
输入 新 密码 和 确认 密码 之 后 ， 单 击 “执行 ”按钮 ， 完 成 对 用 户 密码 的 修改 操作 ， 返 回 主页 面 ， 将 提示 
密码 修改 成 功 。 





3.3 小 结 


本 章 主要 介绍 了 两 个 MySQL 的 图 形 化 管理 工具 , 一 个 是 MySQL 官网 中 提供 的 MySQL Workbench， 
另 一 个 是 使 用 PHP 开发 的 phpMyAdmin, 通过 这 些 图 形 化 管理 软件 可 以 很 方便 地 操作 MySQL 数据 库 ， 
其 中 包括 创建 数据 库 /数据 表 、 管 理 数据 库 /数据 表 ， 以 及 数据 的 导入 和 导出 等 都 是 十 分 方便 的 。 对 于 这 
两 个 图 形 化 管理 工具 ， 读 者 可 根据 自己 的 喜好 来 进行 选择 ， 使 用 哪个 都 可 以 。 


3.4 实践 与 练习 


1. 尝试 使 用 MySQL Workbench 图 形 化 管理 工具 创建 一 个 名 称 为 db_mydatabase 的 数据 库 ， 并 在 
该 数据 库 中 添加 一 名 称 为 tb_user 的 数据 表 。 

2. 尝试 使 用 phpMyAdmin 图 形 化 管理 工具 创建 一 个 名 称 为 db_mydatabase_1 的 数据 库 ， 并 在 该 数 
据 库 中 添加 一 名 称 为 tb_user 的 数据 表 。 
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( 铭 视 频 讲解 : 6 分钟 ) 


启动 并 连接 My5QL 服务 器 后 ， 即 可 对 MySQL 数据 库 进行 操作 ,操作 MySQL 
数据 库 的 方法 非常 简单 。 本 章 中 将 对 操作 My5QL 数据 库 中 的 创建 数据 库 、 修 改 数 
据 库 、 坦 看 数据 库 、 选 择 数 据 库 和 删除 数据 库 进 行 详细 介绍 。 
通过 阅读 本 章 ， 读 者 可 以 : 
| 了 解数 据 库 的 相关 知识 
| 掌握 创建 数据 库 的 方法 
mm 了解 查看 数据 库 的 方法 
MW 了 解 选择 数据 库 的 方法 
MW 掌握 修改 数据 库 的 方法 
MW 掌握 删除 数据 库 的 方法 
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4.1 认识 数据 库 


在 进行 数据 库 操作 前 ， 首 先 需 要 对 其 有 一 个 基本 的 了 解 。 本 节 将 对 数据 库 的 基本 概念 、 数 据 库 对 
象 及 其 相关 知识 进行 详细 的 介绍 。 


4.1.1 数据库 基 本 概念 


数据 库 (DataBase〉 是 按照 数据 结构 来 组 织 、 存 储 和 管理 数据 的 仓库 ， 是 存储 在 一 起 的 相关 数据 
的 集合 。 其 优点 主要 体现 在 以 下 几 方 面 。 

(1) 减少 数据 的 元 余 度 ， 节 省 数据 的 存储 空间 ; 

(2) 具有 较 高 的 数据 独立 性 和 易 扩充 性 ; 

(3) 实现 数据 资源 的 充分 共享 。 

下 面 介绍 与 数据 库 相 关 的 几 个 概念 。 


1. 数据 库 系统 


数据 库 系统 (DataBase System，DBS) 是 采用 数据 库 技术 的 计算 机 系统 ， 是 由 数据 库 数据、 
数据 库 管 理 系 统 ( 软 件 ) 、 数 据 库 管 理 员 (人 员 〉、 硬 件 平台 (硬件 ) 和 软件 平台 (软件 ) 5 部 分 构成 
的 运行 实体 。 其 中 ， 数 据 库 管 理 员 (DataBase Administrator，DBA) 是 对 数据 库 进 行规 划 、 设 计 、 维 
护 和 监视 等 的 专业 管理 人 员 ， 在 数据 库 系统 中 起 着 非常 重要 的 作用 。 


2. 数据 库 管理 系统 


数据 库 管 理 系统 (DataBase Management System，DBMS) 是 数据 库 系 统 的 一 个 重要 组 成 部 分 ， 是 
位 于 用 户 与 操作 之 问 的 一 层 数 据 管理 软件 ， 负 责 数据 库 中 的 数据 组 织 、 数 据 操纵 、 数 据 维护 和 数据 服 
务 等 。 主 要 具有 如 下 功能 。 

(1) 数据 存 取 的 物理 构建 : 为 数据 模式 的 物理 存 取 与 构建 提供 有 效 的 存 取 方 法 与 手段 。 

(2) 数据 操纵 功能 : 为 用 户 使 用 数据 库 的 数据 提供 方便 ， 如 查询 、 插 入 、 修 改 、 删 除 等 以 及 简单 
的 算术 运算 和 统计 。 

(3) 数据 定义 功能 : 用 户 可 以 通过 数据 库 管 理 系统 提供 的 数据 定义 语言 (Data Definition 
Language，DDL) 方便 地 对 数据 库 中 的 对 象 进行 定义 。 

(4) 数据 库 的 运行 管理 : 数据 库 管 理 系统 统一 管理 数据 库 的 运行 和 维护 ， 以 保障 数据 的 安全 性 、 
完整 性 、 并 发 性 和 故障 的 系统 恢复 性 。 

(5) 数据 库 的 建立 和 维护 功能 : 数据 库 管 理 系统 能 够 完成 初始 数据 的 输入 和 转换 、 数 据 库 的 转 储 
和 恢复 、 数 据 库 的 性 能 监视 和 分 析 等 任务 。 


3. 关系 数据 库 
关系 数据 库 是 支持 关系 模型 的 数据 库 。 关 系 模型 由 关系 数据 结构 、 关 系 操作 集合 和 完整 性 约束 3 
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部 分 组 成 。 

(1) 关系 数据 结构 : 在 关系 模型 中 数据 结构 单一 ， 现 实 世 界 的 实体 以 及 实体 间 的 联系 均 用 关系 来 
表示 ， 实 际 上 关系 模型 中 数据 结构 就 是 一 张 二 维 表 。 

(2) 关系 操作 集合 : 关系 操作 分 为 关系 代数 、 关 系 演算 、 具 有 关系 代数 和 关系 演算 双重 特点 的 语 
言 (SQL) 。 

(3) 完整 性 约束 : 完整 性 约束 包括 实体 完整 性 、 参 照 完 整 性 和 用 户 定义 完整 性 。 





4.1.2 数据库 常 用 对 象 


在 MySQL 的 数据 库 中 ， 表 、 视 图 、 存 储 过 程 和 索引 等 具体 存储 数据 或 对 数据 进行 操作 的 实体 都 
被 称 为 数据 库 对 象 。 下 面 介绍 几 种 常用 的 数据 库 对 象 。 


1. 表 

表 是 包含 数据 库 中 所 有 数据 的 数据 库 对 象 ， 由 行 和 列 组 成 ， 用 于 组 织 和 存储 数据 。 

2. 字段 

表 中 每 列 称 为 一 个 字段 ， 字 段 具 有 自己 的 属性 ， 如 字段 类 型 、 字 段 大 小 等 。 其 中 ， 字 段 类 型 是 字 
段 最 重要 的 属性 ， 它 决定 了 字段 能 够 存储 哪 种 数据 。 

SQL 规范 支持 5 种 基本 字段 类 型 : 字符 型 、 文 本 型 、 数 值 型 、 罗 辑 型 和 日 期 时 间 型 。 

3. 索引 


索引 是 一 个 单独 的 、 物 理 的 数据 库 结 构 。 它 是 依赖 于 表 建 立 的 ， 在 数据 库 中 索引 使 数据 库 程序 无 
须 对 整个 表 进 行 扫描 ， 就 可 以 在 其 中 找到 所 需 的 数据 。 


4. 视图 

视图 是 从 一 张 或 多 张 表 中 导出 的 表 〈 也 称 虚拟 表 ) ， 是 用 户 查看 数据 表 中 数据 的 一 种 方式 。 表 中 
包括 几 个 被 定义 的 数据 列 与 数据 行 ， 其 结构 和 数据 建立 在 对 表 的 查询 基础 之 上 。 

5. 存储 过 程 

存储 过 程 (Stored Procedure) 是 一 组 为 了 完成 特定 功能 的 SQL 语句 集合 (包含 查询 、 插 入 、 删 除 


和 更 新 等 操作 ) ， 经 编译 后 以 名 称 的 形式 存储 在 SQL Server 服务 器 端的 数据 库 中 ， 由 用 户 通过 指定 存 
储 过 程 的 名 字 来 执行 。 当 这 个 存储 过 程 被 调用 执行 时 ， 这 些 操作 也 会 同时 执行 。 





4.1.3 ”系统 数据 库 


系统 数据 库 是 指 安装 完 MySQL 服务 器 后 ， 会 附带 一 些 数据 库 。 例 如 ， 在 默认 安装 的 MySQL 
Workbench 中 , 会 默认 创建 如 图 4.1 所 示 的 5 个 数据 库 , 这 些 数据 库 就 称 为 系统 数据 库 。 系 统 数据 库 会 
记录 一 些 必需 的 信息 ， 用 户 是 不 能 直接 修改 这 些 系统 数据 库 的 。test 和 sakila 除外 ， 这 两 个 数据 库 中 的 
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信息 对 于 系统 不 是 必需 的 ， 可 以 进行 修改 。 下 面 将 对 图 4.1 中 所 列 的 系统 数据 库 分 别 进 行 介 绍 。 
1. information_schema 数据 库 


information_schema 数据 库 主要 用 于 存储 数据 库 对 象 的 相关 信息 。 例 如 ， 用 户 表 信息 、 列 信息 、 权 
限 信息 、 字 符 集 信息 和 分 区 信息 等 。 


2. performance_schema 数据 库 
performance_schema 数据 库 主 要 用 于 存储 数据 库 服务 器 性 能 参数 。 
3. sakila 数据 库 


sakila 数据 库 是 MySQL 提供 的 样 例 数据 库 。 该 数据 库 共 有 16 张 数据 表 ， 这 些 数据 表 都 是 比较 常 
见 的 ， 在 设计 数据 库 时 ， 可 以 参照 这 些 样 例 数据 表 来 快速 完成 所 需 的 数据 表 。 


4. test 数据 库 


test 数据 库 是 MySQL 数据 库 管 理 系统 自动 创建 的 测试 数据 库 ， 该 数据 库 中 没有 创建 任何 数据 表 ， 
对 于 任何 用 户 都 可 以 使 用 这 个 数据 库 。 一 般 情 况 下 ， 不 建议 直接 使 用 该 数据 库 。 


5. world 数据 库 


world 数据 库 是 MySQL 数据 库 管 理 系统 自动 创建 的 数据 库 ， 该 数据 库 中 只 包括 3 张 数据 表 ， 分 别 
保存 城市 、 国 家 和 国家 使 用 的 语言 等 内 容 。 


/. 
说明 
在 MySQL 中 ,使 用 最 多 的 是 用 户 数据 库 ， 也 就 是 用 户 根据 实际 需求 所 创建 的 数据 库 ， 例 如 在 
图 4.2 中 的 db_mrsoft 就 是 我 们 创建 的 用 户 数据 库 。 





SCHEMAS 


Q Fiter objects 


infoxmation_schema 
perfoxmance_schema 


pb information_schema 
performance_schema 


目 test 


> 
bp sakila 
p 
p world 





图 4.1 系统 数据 库 图 4.2 用 户 数 据 库 


4.2 创建 数据 库 


在 MySQL 中 ， 可 以 使 用 CREATE DATABASE 语句 和 CREATE SCHEMA 语句 创建 MySQL 数据 
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库 ， 其 语法 如 下 。 
CREATE {DATABASEISCHEMA} [IF NOT EXISTS] 数据 库 名 


IDEFAULT] CHARACTER SET [=] 字符 集 | 
[DEFAULT] COLLATE [=] 校对 规则 名 称 


E 
Do 


在 语法 中 ， 花 括号 “{}” 表 示 必 选项 ; 中 括号 “[]” 表 示 可 选项 ; 竖 线 “|” 表 示 分 隔 符 两 侧 的 
内 容 为 “或 ”的 关系 。 在 上 面 的 语法 中 , {DATABASEISCHEMA} 表 示 要 么 使 用 关键 字 DATABASE， 


要 么 使 用 SCHEMA， 但 不 能 全 不 使 用 。 


参数 说 明 如 下 。 

(1) [IF NOT EXISTS]: 可 选项 ， 表 示 在 创建 数据 库 前 进 
行 判断 ， 只 有 该 数据 库 目前 尚未 存在 时 才 执 行 创建 语句 。 

(2) 数据 库 名 : 必须 指定 的 ， 在 文件 系统 中 ，MySQL 的 
数据 存储 区 将 以 目录 方式 表示 MySQL 数据 库 。 因此, 这 里 的 数 
据 库 名 必须 符合 操作 系统 文件 夹 的 命名 规则 , 而 在 MySQL 中 是 
不 区 分 大 小 写 的 。 

(3) [DEFAULT]: 可 选项 ， 表 示 指 定 默认 值 。 

(4) CHARACTER SET [=] 字符 集 : 可 选项 ， 用 于 指定 
数据 库 的 字符 集 。 如 果 不 想 指定 数据 库 所 使 用 的 字符 集 ， 那 么 
就 可 以 不 使 用 该 项 ， 这 时 MySQL 会 根据 MySQL 服务 器 默认 
使 用 的 字符 集 来 创建 该 数据 库 。 这 里 的 字符 集 可 以 是 GB2312 
或 者 GBK (简体 中 文 ) 、UTF8 (针对 Unicode 的 可 变 长 度 的 
字符 编码 ， 也 称 万 国 码 ) 、BIG5 (繁体 中 文 ) 、Latinl (拉丁 
文 ) 等 。 其 中 最 常用 的 就 是 UTF8 和 GBK。 

(5) COLLATE [=] 校对 规则 名 称 : 可 选项 ， 用 于 指定 字 
符 集 的 校对 规则 。 例 如 ，utf8_bin 或 者 gbk_chinese_ci。 具 体 都 
有 哪些 校对 规则 ， 可 以 在 MySQL 的 图 形 化 工具 Workbench 的 
创建 数据 库 的 窗口 中 找到 ， 如 图 4.3 所 示 。 

在 创建 数据 库 时 ， 数 据 库 命名 有 以 下 几 项 规则 。 

(1) 不 能 与 其 他 数据 库 重 名 ， 否 则 将 发 生 错 误 。 











(2) 名 称 可 以 由 任意 字母 、 阿 拉 伯 数字 、 下 划 线 (_) 和 “$” 组 成 ， 可 以 使 用 上 述 的 任意 字符 开 


头 ， 但 不 能 使 用 单独 的 数字 ， 和 否则 会 造成 它 与 数值 相 混淆 。 


(3) 名 称 最 长 可 为 64 个 字符 ， 而 别名 最 多 可 长 达 256 个 字符 。 


(4) 不 能 使 用 MySQL 关键 字 作 为 数据 库 名 、 表 名 。 


(5) 默认 情况 下 , 在 Windows 下 数据 库 名 、 表 名 的 大 小 写 是 不 敏感 的 , 而 在 Linux 下 数据 库 名 、 


gb ollation 
gb2312 - default collati 
gb2312 - gb2312_cdhinese_G 
gb2312 - gb2312_bin 
greek - default collation 
greek - greek_general_G 
greek - greek_bin 
中 1250 - default colation 
中 1250 - cp1250_general_d 
p1250 - cp1250_czech_cs 
中 1250 - cp1250_croatian_d 
中 1250 - cp1250_bin 
gbk - default collation 
gbk - gbk_chinese_ci 
gbk -gbk bn 
latin5 - default collation 
latin5 -latn5_turkish_G 
latin5 -latin5_bin 
armsci8 - default collation 
armsci8 - armsci8_general_d 
armsciS - armscis bin 
tf8 - default collation 
-ui 
-utf8_czech_d 
-utf8_danish_o 
-utf8_esperanto_G 
-utf8_estonian_G 
-utf8_general_d 
-utfs_general_mysql500 


utf8 n_d 
-utfs -icelandic_G 
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表 名 的 大 小 写 是 敏感 的 。 为 了 便于 数据 库 在 平台 间 进 行 移植 ， 建 议 读者 采用 小 写 来 定义 数据 库 名 和 
表 名 。 


4.2.1 通过 CREATE DATABASE 语句 创建 基本 数据 库 


例 4.1 通过 CREATE DATABASE 语句 创建 一 个 名 称 为 db_admin 的 数据 库 , 具体 代码 如 下 。( 实 
例 位 置 ， 光盘 \TMNsI\4\4.1) 


create database db_admin; 


运行 效果 如 图 4.4 所 示 。 


nysql> create database db_adnin; 
Query OK, 1 row affected C0 
ET 


机 @ 创建 数据 库 成 功 





图 4.4 通过 CREATE DATABASE 语句 创建 MySQL 数据 库 


4.2.2 ”通过 CREATE SCHEMA 语句 创建 基本 数据 库 


上 面 介绍 的 例 4.1 是 最 基本 的 创建 数据 库 的 方法 ， 实 际 上 ， 还 可 以 通过 语法 中 给 出 的 CREATE 
SCHEMA 来 创建 数据 库 ， 两 者 的 功能 是 一 样 的 。 在 使 用 MySQL 官网 中 提供 的 MySQL Workbench 图 
形 化 工具 创建 数据 库 时 ， 使 用 的 就 是 这 种 方法 。 

例 4.2 通过 CREATE SCHEMA 语句 创建 一 个 名 称 为 db_adminl 的 数据 库 ， 上 有 具体 代码 如 下 。( 实 
例 位 置 ， 光 盘 \TMNsI\4\4.2) 


create schema db_admin1; 
运行 效果 如 图 4.5 所 示 。 


》 create schena db_adnin1; 
OK。 1 row affec 


Inysal 创建 数据 库 db_admin 
@ 创建 数据 库 成 功 


图 4.5 通过 CREATE SCHEMA 语句 创建 MySQL 数据 库 





4.2.3 创建 指定 字符 集 的 数据 库 











在 创建 数据 库 时 ， 如 果 不 指定 其 使 用 的 字符 集 或 者 是 字符 集 的 校对 规则 ， 那 么 将 根据 my.ini 文件 
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中 指定 的 default-character-set 变量 的 值 来 设置 其 使 用 的 字符 集 。 从 创建 数据 库 的 基本 语法 中 可 以 看 出 ， 
在 创建 数据 库 时 ， 还 可 以 指定 数据 库 所 使 用 的 字符 集 ， 下 面 将 通过 一 个 具体 的 例子 来 演示 如 何在 创建 
数据 库 时 ， 指 定 字符 集 。 

例 4.3 通过 CREATE DATABASE 语句 创建 一 个 名 称 为 db_test 的 数据 库 ， 并 指定 其 字符 集 为 
GBK， 有 具体 代 码 如 下 。《〔 实 例 位 置 : 光盘 \TMNsI\4\4.3) 


CREATE DATABASE db _test 
CHARACTER SET = GBK; 








运行 效果 如 图 4.6 所 示 。 


mysql> CREATE DATABASE db test 
> CHARACTER SET GBK; 
Query OK, 1 rom affected (0.83 sec)> 


设置 字符 集 为 GBK 





nysgl> 


图 4.6 创建 使 用 GBK 字符 集 的 MySQL 数据 库 
4.2.4 ”创建 数据 库 前 判断 是 否 存在 同名 数据 库 


在 MySQL 中 ， 不 允许 同一 系统 中 存在 两 个 相同 名 称 的 数据 库 ， 如 果 要 创建 的 数据 库 名 称 已 经 存 
在 ， 那 么 系统 将 给 出 以 下 错误 信息 。 

ERROR 1007 (HY000): Can' create database 'db_test'; database exists 

为 了 避免 错误 的 发 生 ， 在 创建 数据 库 时 ， 可 以 使 用 正 NOT EXISTS 选项 来 实现 在 创建 数据 库 前 判 
断 该 数据 库 是 否 存在 ， 只 有 不 存在 时 才 会 进行 创建 。 

例 4.4 通过 CREATE DATABASE 语句 创建 一 个 名 称 为 db_testl 的 数据 库 , 并 在 创建 前 判断 该 数 

居 库 名 称 是 否 存在 ， 只 有 不 存在 时 才 进 行 创建 ， 有 具体 代码 如 下 。《【〈 实 例 位 置 : 光盘 \TMNsI\4\4.4) 
CREATE DATABASE IF NOT EXISTS db_test1; 





运行 效果 如 图 4.7 所 示 。 


Inysql> CREATE DATABASE IF NOT E 


Query OK, 1 row a ed 《0.61 
下 创建 数据 库 前 判断 其 是 否 存在 
图 4.7 创建 数据 库 前 判断 是 否 存在 同名 数据 库 
再 次 执行 上 面 的 语句 ， 将 不 再 创建 数据 库 db_testl， 显 示 效 果 如 图 4.8 所 示 。 








mysql> CREATE DATABASE IF NOT EX S db_testi; 
Query OK, 1 row affected, | 8.88 sec> 


显示 有 一 个 警告 ,数据库 没 有 创建 





nysql> 


图 4.8 创建 已 经 存在 的 数据 库 的 效果 
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4.3 查看 数据 库 


成 功 创建 数据 库 后 ， 可 以 使 用 SHOW 命令 查看 MySQL 服务 器 中 的 所 有 数据 库 信 息 ， 语 法 如 下 。 
SHOW {DATABASESISCHEMAS} 
[LIKE ' 模 式 ' WHERE 条 件 ] 


参数 说 明 如 下 。 

(1) {DATABASESISCHEMAS}: 表示 必须 有 一 个 是 必 选 项 ， 用 于 列 出 当前 用 户 权限 范围 内 所 能 
查看 到 的 所 有 数据 库 名 称 。 这 两 个 选项 的 结果 是 一 样 的 ， 使 用 哪个 都 可 以 。 

(2) LIKE: 可 选项 ， 用 于 指定 匹配 模式 。 

(3) WHERE: 可 选项 ， 用 于 指定 数据 库 名 称 查询 范围 的 条 件 。 

例 4.5 在 4.2.1 节 中 创建 了 数据 库 db_admin， 下 面 使 用 SHOW DATABASES 语句 查看 MySQL 
服务 器 中 的 所 有 数据 库 名 称 ， 代 码 如 下 。《 实 例 位 置 ， 光盘 \TMsl4\4.5) 


SHOW DATABASES; 
运行 结果 如 图 4.9 所 示 。 


@ 查看 所 有 数据 库 


1 dh_adnin : 
De @ 显示 数据 库 列表 


! perfornance_schen 


1 test 


《0.80 sec) 





图 4.9 查看 数据 库 
从 图 4.9 运行 的 结果 可 以 看 出 ， 通 过 SHOW 命令 查看 MySQL 服务 器 中 的 所 有 数据 库 ， 结 果 显 示 
MySQL 服务 器 中 有 8 个 数据 库 ， 这 8 个 数据 库 包 括 系 统 数 据 库 。 
如 果 MySQL 服务 器 中 的 数据 库 比较 多 ， 也 可 以 通过 指定 匹配 模式 来 筛选 想 要 得 到 的 数据 库 ， 下 
面 将 通过 一 个 具体 的 实例 来 演示 如 何 通过 LIKE 关键 字 筛 选 要 查看 的 数据 库 。 
例 4.6 筛选 以 db 开头 的 数据 库 名 称 ， 代 码 如 下 。《〔 实 例 位 置 ， 光盘 \IMNsI\4\4.6) 
SHOW DATABASES LIKE 'db_%'; 
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SHOU DATABASES LIKE ’db x’; 


1 Database db x» 
@ 查看 以 db 开头 的 数据 库 
adnin 
adnmnini 
test 


test1 四 显示 4 个 查询 结 


ous in set (8.BL sec》 





nysql> 


图 4.10 第 选 以 db 开头 的 数据 库 名 称 
4.4 选择 数据 库 


在 MySQL 中 , 使 用 CREATE DATABASE 语句 创建 数据 库 后 , 该 数据 库 并 不 会 自动 成 为 当前 数据 
库 。 如 果 想 让 它 成 为 当前 数据 库 ， 需 要 使 用 MySQL 提供 的 USE 语句 来 实现 ，USE 语句 可 以 实现 选择 
-个 数据 库 ， 使 其 成 为 当前 数据 库 。 只 有 使 用 USE 语句 指定 某 个 数据 库 为 当前 数据 库 后 ， 才 能 对 该 数 
据 库 及 其 存储 的 数据 对 象 执行 操作 。USE 语句 的 语法 格式 如 下 。 
USE 数据库 名 ; 


说明 


使 用 USE 语句 将 数据 库 指 定 为 当前 数据 库 后 ， 当 前 数据 库 在 当前 工作 会 话 关 闭 ( 即 断 开 与 该 
数据 库 的 连接 ) 或 再 次 使 用 USE 语句 指定 数据 库 时 ， 结 束 工作 状态 。 


例 4.7 选择 名 称 为 db_admin 的 数据 库 ， 设 置 其 为 当前 默认 的 数据 库 ， 有 具体 代码 如 下 。《〔 实 例 位 
置 光盘 \TMNsI\M4\4.7) 

USE db_admin; 

执行 结果 如 图 4.11 所 示 。 


nysql> use db_adnin; 


Database changed @ 选择 指定 的 数据 库 


mysql> 





四 成 功 地 选择 了 该 数据 库 











4.11 选择 数据 库 
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4.5 修改 数据 库 


在 MySQL 中 ， 创 建 一 个 数据 库 后 ， 还 可 以 对 其 进行 修改 ， 不 过 这 里 的 修改 是 指 可 以 修改 被 创建 
数据 库 的 相关 参数 ， 并 不 能 修改 数据 库 名 。 修 改 数据 库 名 不 能 使 用 这 个 语句 。 修 改 数据 库 可 以 使 
ALTER DATABASE 或 者 ALTER SCHEMA 语句 来 实现 。 修 改 数据 库 的 语句 的 语法 格式 如 下 。 

ALTER {DATABASE | SCHEMA} [数据 库 名 ] 


IDEFAULT] CHARACTER SET [=] 字符 集 
| [DEFAULT] COLLATER [=] 校对 规则 名 称 


参数 说 明 如 下 。 

(1) {DATABASESISCHEMAS}: 表示 必须 有 一 个 是 必 选 项 ， 这 两 个 选项 的 结果 是 一 样 的 ， 使 用 
哪个 都 可 以 。 

(2) [数据 库 名 ]: 可 选项 ， 如 果 不 指定 要 修改 的 数据 库 ， 那 么 将 表示 修改 当前 默认) 的 数据 库 。 

(3) [DEFAULT]: 可 选项 ， 表 示 指 定 默认 值 。 

(4) CHARACTER SET [=] 字符 集 : 可 选项 ， 用 于 指定 数据 库 的 字符 集 。 如 果 不 想 指定 数据 库 
所 使 用 的 字符 集 ， 那 么 就 可 以 不 使 用 该 项 ， 这 时 MySQL 会 根据 MySQL 服务 器 默认 使 用 的 字符 集 来 
创建 该 数据 库 。 这 里 的 字符 集 可 以 是 GB2312 或 者 GBK (简体 中 文 ) 、UTF8 (针对 Unicode 的 可 变 
长 度 的 字符 编码 ， 也 称 万 国 码 ) 、BIG5 (繁体 中 文 ) 、Latinl (拉丁 文 ) 等 。 其 中 最 常用 的 就 是 UTF8 
和 GBK。 

(5) COLLATE [=] 校对 规则 名 称 : 可 选项 ， 用 于 指定 字符 集 的 校对 规则 。 例 如 ，utf8_bin 或 者 
gbk_chinese_ci。 这 与 4.2 节 创 建 数据 库 的 语法 中 的 该 从 句 是 相同 的 。 


人 注意 
在 使 用 ALTER DATABASE 或 者 ALTER SCHEMA 语句 时 ， 用 户 必须 具有 对 数据 库 进行 修改 
的 权限 。 

















例 4.8 修改 例 4.1 中 创建 的 数据 库 do_admin， 设 置 默 认 字 符 集 和 校对 规则 ， 具 体 代码 如 下 。( 实 
例 位 置 : 光盘 \TMNsI\4\4.8) 


ALTER DATABASE db_admin 
DEFAULT CHARACTER SET gbk 
DEFAULT COLLATE gbk_chinese_ci; 


执行 结果 如 图 4.12 所 示 。 
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Imnysul> ALIER DATABASE db @ 修改 字符 集 为 GBK 


@ 修改 字符 集 的 校对 规则 为 简体 中 文 





图 4.12 设置 默认 字符 集 和 校对 规则 
4.6 删除 数据 库 


在 MySQL 中 ， 可 以 通过 使 用 DROP DATABASE 语句 或 者 DROP SCHEMA 语句 来 删除 已 经 存在 
的 数据 库 。 使 用 该 命令 删除 数据 库 的 同时 ， 该 数据 库 中 的 表 ， 以 及 表 中 的 数据 也 将 永久 删除 ， 因 此 ， 
在 使 用 该 语句 删除 数据 库 时 一 定 要 小 心 ， 以 免 误 删除 有 用 的 数据 库 。DROP DATABASE 或 者 DROP 
SCHEMA 语句 的 语法 格式 如 下 。 
DROP {DATABASE|SCHEMA} [IF EXISTS] 数据 库 名 ; 
参数 说 明 如 下 。 
(1) {DATABASESISCHEMAS}: 表示 必须 有 一 个 是 必 选 项 ， 这 两 个 选项 的 结果 是 一 样 的 ， 使 用 
哪个 都 可 以 。 
(2) [IF EXISTS]: 用 于 指定 在 删除 数据 前 ， 先 判断 该 数据 库 是 否 已 经 存在 ， 只 有 已 经 存在 时 ， 才 
会 执行 删除 操作 ， 这 样 可 以 避免 删除 不 存在 的 数据 库 时 ， 产 生 异 常 。 
$b 注意 
在 使 用 DROP DATABASE 或 者 DROP SCHEMA 语句 时 ， 用 户 必须 具有 对 数据 库 进行 删除 的 
权限 。 











人 ie 注意 
在 删除 数据 库 时 ， 该 数据 库 上 的 用 户 权限 是 不 会 自动 被 删除 的 . 


息 s 注 意 
« 

删除 数据 库 的 操作 应 该 谨慎 使 用 , 一旦 执行 该 操作 ， 数据库 的 所 有 结构 和 数据 都 会 被 删除 ， 没 
有 恢复 的 可 能 ， 除 非 数据 库 有 备份 。 


例 4.9 通过 DROP DATABASE 语句 删除 名 为 db_admin 的 数据 库 ， 具 体 代码 如 下 。 《实例 位 置 ; 
光盘 \TMsl\4\4.9) 
DROP DATABASE db_admin; 


执行 效果 如 图 4.13 所 示 。 
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Inysql> drop database db_adnin; 
Query OK, @ rows affected (8.83 sec) 


nysql> @ 成 功 删 除 该 数据 


图 4.13 ”删除 数据 库 
当 使 用 上 面 的 命令 删除 数据 库 时 ， 如 果 指 定 的 数据 库 不 存在 ， 将 产生 如 图 4.14 所 示 的 异常 信息 。 





@ 删除 数据 库 顾 db_admin 





Inysql> DROP DATABASE db_111; 


IERROR 1998 〈《HYBBB>: Can’t drop database ’db_111’; database doesn’t exist 
nysql> - 





图 4.14 删除 不 存在 的 数据 库 出 错 
为 了 解决 这 一 问题 ， 可 以 在 DROP DATABASE 语句 中 使 用 正 EXISTS 从 句 来 保证 只 有 当 数 据 库 
存在 时 才 执 行 删除 数据 库 的 操作 。 下 面 通过 一 个 具体 的 例子 来 演示 这 一 功能 。 
例 4.10 通过 DROP DATABASE 语句 删除 名 称 为 db_111 的 数据 库 〈 该 数据 库 不 存在 ) ， 有 具体 代 
码 如 下 。 《实例 位 置 : 光盘 \TM'sI\4\4.10) 


SHOW DATABASES LIKE 'db_%'; 
DROP DATABASE IF EXISTS db_111; 


执行 效果 如 图 4.15 所 示 。 


mysql> SHOW DATABASES LIKE "dh 
1 Database Cdb x> ! © 查看 以 db_ 开 头 的 数据 库 


db_admin 


@ 删除 不 存在 的 数据 库 db_111, 并 没有 出 错 


DROP DATABA 
y OK, @ rows aff 


nysql> ,~ 





图 4.15 ”删除 不 存在 的 数据 库 未 出 错 


¢ Ms 注意 

MySQL 安装 后 ,系统 会 自动 创建 两 个 名 称 分 别 为 performance_ schema 和 mysql 的 系统 数据 库 ， 
MySQL 把 与 数据 库 相 关 的 信息 存储 在 这 两 个 系统 数据 库 中 ， 如 果 删 除了 这 两 个 数据 库 ， 那 么 
MySQL 将 不 能 正常 工作 ， 所 以 这 两 个 数据 库 一 定 不 能 删除 。 
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4.7 小 结 


本 章 首 先 介绍 了 数据 库 的 基本 概念 、 数 据 库 的 常用 对 象 ， 以 及 MySQL 中 的 系统 数据 库 ， 然 后 介 
绍 了 如 何 修改 数据 库 、 查 看 数据 库 、 选 择 数据 库 、 修 改 数据 库 和 删除 数据 库 。 其 中 ， 创 建 数据 库 、 选 
择 数 据 库 和 删除 数据 库 需 要 重点 掌握 ， 在 实际 开发 中 经 常会 应 用 到 。 


4.8 实践 与 练习 


1. 通过 CREATE SCHEMA 语句 创建 一 个 名 称 为 db_mr 的 数据 库 ， 并 指定 其 字符 集 为 UTF8。 ( 答 
案 位 置 ， 光盘 \TMNsl4\4.11) 

2. 通过 DROP SCHEMA 语句 删除 第 1 题 中 创建 的 数据 库 db_mr， 并 且 指 定 只 有 该 数据 库存 在 时 
才 删 除 。〔 答 案 位 置 ， 光盘 \TMNs1\4\4.12) 

3. 通过 SHOW SCHEMAS 语句 筛选 以 db_ 开 头 的 数据 库 名 称 。〔 答 案 位 置 ， 光 盘 \TMNsI\4\4.13) 
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( 盈 视频 讲解 ; 12 分 钟 ) 


使 用 存储 引擎 可 以 加 快 查 询 的 速度 ， 并 且 每 一 种 引擎 都 存在 不 同 的 含义 。 
My5QL 的 数据 类 型 是 数据 的 一 种 属性 ， 其 可 以 决定 数据 的 存储 格式 、 有 效 范 围 和 
相应 的 限制 ， 并 且 可 以 让 读者 了 解 如 何 选 择 合适 的 数据 类 型 。 本 章 将 对 My5QL 的 
存储 引擎 和 数据 类 型 的 使 用 进行 详细 的 讲解 。 

通过 阅读 本 章 ， 读 者 可 以 : 

了 解 MySQL 存储 引擎 

了 解 并 查询 MySQL 中 支持 的 存储 引擎 
掌握 选择 存储 引擎 的 方法 

掌握 设置 数据 表 的 存储 引擎 的 方法 
掌握 MySQL 的 数据 类 型 


于 于 于 至 至 


第 5 章 存储 引擎 及 数据 类 型 


5.1 MySQL 存储 引擎 


存储 引擎 其 实 就 是 如 何 存储 数据 、 如 何 为 存储 的 数据 建立 索引 和 如 何 更 新 、 查 询 数据 等 技术 的 实 
现 方法 。 因 为 在 关系 数据 库 中 数据 是 以 表 的 形式 存储 的 ， 所 以 存储 引擎 也 可 以 称 为 表 类 型 〈 即 存储 和 
操作 此 表 的 类 型 ) 。 在 Oracle 和 SQL Server 等 数据 库 中 只 有 一 种 存储 引擎 ， 所 有 数据 存储 管理 机 制 都 
是 一 样 的 ;而 MySQL 数据 库 提 供 了 多 种 存储 引擎 。 用 户 可 以 根据 不 同 的 需求 为 数据 表 选 择 不 同 的 存 
储 引擎 ， 用 户 也 可 以 根据 需要 编写 自己 的 存储 引擎 。 


5.1.1 MySQL 存储 引擎 的 概念 


MySQL 中 的 数据 用 各 种 不 同 的 技术 存储 在 文件 〈 或 者 内 存 ) 中。 这 些 技术 中 的 每 一 种 技术 都 使 用 
不 同 的 存储 机 制 、 索 引 技巧 、 锁 定 水 平 并 且 最 终 提供 广泛 的 、 不 同 的 功能 和 能 力 。 通 过 选择 不 同 的 技 
术 ， 能 够 获得 额外 的 速度 或 者 功能 ， 从 而 改善 应 用 的 整体 功能 。 

这 些 不 同 的 技术 以 及 配套 的 相关 功能 在 MySQL 中 被 称 作 存 储 引擎 〈 也 称 作 表 类 型 ) 。MySQL 默 
认 配 置 了 许多 不 同 的 存储 引擎 ， 可 以 预先 设置 或 者 在 MySQL 服务 器 中 启用 。 可 以 选择 适用 于 服务 器 、 
数据 库 和 表格 的 存储 引擎 ， 以 便 在 选择 如 何 存 储 信 息 、 如 何 检索 这 些 信息 以 及 需要 的 数据 结合 什么 性 
能 和 功能 的 时 候 为 其 提供 最 大 的 灵活 性 。 


5.1.2 查询 MySQL 中 支持 的 存储 引擎 


1. 查询 支持 的 全 部 存储 引擎 
在 MySQL 中 ， 可 以 使 用 SHOW ENGINES 语句 查询 MySQL 中 支持 的 存储 引擎 。 其 查询 语句 
如 下 。 

SHOW ENGINES; 

SHOW ENGINES 语句 可 以 用 “;” 结 束 ， 也 可 以 用 “\g” 或 者 “\G” 结 束 。“\g” 与 “;” 的 作 
是 相同 的 ，“\G” 可 以 让 结果 显示 得 更 加 美观 。 

使 用 SHOW ENGINES \g 语句 查询 的 结果 如 图 5.1 所 示 。 




















Inysql> show engine: 


Default engine of MySQL 3.23 with great perfornance 
! NO ! 
U storage engine 
! NO ! NO ! 
SG_MYISRM ! YES ollection of identical MyISAM tabl 
! NO NO !NO ! 
KHOLE ! YES /dev/null storage engine Canything you write to it disa 
> ! NO NO !NO ! 
EDERRTED ! Federated MySQL storage engine 
! NULL NULL ! NULL ! 
InnoDB ! DEFRULT ! Supports transactions, row-level locking, and foreign hk 
! YES YES ! YES ! 
ARCHIUE YES Archive storage engine 
! NO NO !NO ! 
MEMORY YES ored in memory。 useful for temporary tabl 


ley: 








图 5.1 使 用 SHOW ENGINES \G 语句 查 


结果 如 | 


询 MySQL 中 支持 的 存储 引擎 





使 用 SHOW ENGINES \G 语句 查询 的 5.2 所 示 























国 MysQL56 command Line Cn 让 呈 刁 EE 
engine 
Savepo 
Engine: MRG_MYISAM | 
pport: YES ' 
Comment: Collection of identical MylSAM tables 用 
NO 
: NO MH 
| No | 
Engine: MyISAM 中 
Support: YES 
omnent: MyISRM storage engine 
actions: NO | 
A: NO | 
avepoints: NO 
vepoin | 
Engine: BLACKHOLE 册 
port: YES 
Comment: /dev/null storage engine 《anything you write to it disappear | 
由 
#8: NO 
Savepoint: NO 
Engine: CSU - 
' 
一 j 





图 5.2 使 用 SHOW ENGINES \G 语句 查询 MySQL 中 支持 的 存储 引擎 
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查询 结果 中 的 Engine 参数 指 的 是 存储 引擎 的 名 称 ; Support 参数 指 的 是 MySQL 是 否 支 持 该 类 引擎 ， 
YES 表示 支持 ; Comment 参数 指 对 该 引擎 的 评论 。 

从 查询 结果 中 可 以 看 出 ，MySQL 支持 多 个 存储 引擎 ， 其 中 InnoDB 为 默认 存储 引擎 。 

2. 查询 默认 的 存储 引擎 

如 果 想 要 知道 当前 MySQL 服务 器 采用 的 默认 存储 引擎 是 什么 ,可 以 通过 执行 SHOW VARIABLES 
命令 来 查看 。 查 询 默认 的 存储 引擎 的 SQL 语句 如 下 。 

SHOW VARIABLES LIKE 'storage_engine%'; 

例 5.1 查询 默认 的 存储 引擎 ， 有 具体 代码 如 下 。《〈 实 例 位 置 : 光盘 \TMNsINS\S.1) 

SHOW VARIABLES LIKE 'storage_engine%'; 

执行 效果 如 图 5.3 所 示 。 








HOW UARIABLES LIKE ’storage_enginex’; 


alue 


1 storage_engine ! InnoDB ! 
+ + 


1 row in set (80.00 sec) 


Inysql> 。 





图 5.3 查询 默认 的 存储 引擎 


从 图 5.3 中 可 以 看 出 ， 当 前 MySQL 服务 器 采用 的 默认 存储 引擎 是 InnoDB。 

有 些 表 根 本 不 用 来 存储 长 期 数据 ， 实 际 上 用 户 需 要 完全 在 服务 器 的 RAM 或 特殊 的 临时 文件 中 创 
建 和 维护 这 些 数据 ， 以 确保 高 性 能 ， 但 这 样 也 存在 很 高 的 不 稳定 风险 。 还 有 一 些 表 只 是 为 了 简化 对 一 
组 相同 表 的 维护 和 访问 ， 为 同时 与 所 有 这 些 表 交互 提供 一 个 单一 接口 。 另 外 ， 还 有 其 他 一 些 特别 用 途 
的 表 ， 但 重点 是 : MySQL 支持 很 多 类 型 的 表 ， 每 种 类 型 都 有 自己 特定 的 作用 、 优 点 和 缺点 。 MySQL 
还 相应 地 提供 了 很 多 不 同 的 存储 引擎 ， 可 以 以 最 适合 于 应 用 需求 的 方式 存储 数据 。 MySQL 有 多 个 可 用 
的 存储 引擎 ， 下 面 主要 介绍 IhnoDB、MyYISAM 和 MEMORY3 种 存储 引擎 。 


5.1.3 InnoDB 存储 引擎 


InnoDB 已 经 开发 了 十 余年 ,遵循 CNU 通 用 公开 许可 (GPL ) 发 行 .InnoDB 已 经 被 一 些 重量 级 Intemet 
公司 所 采用 ， 如 雅虎 、Slashdot 和 Google， 为 用 户 操作 非常 大 的 数据 库 提 供 了 一 个 强大 的 解决 方案 。 
InnoDB 给 MySQL 的 表 提供 了 事务 、 回 滚 、 骨 溃 修 复 能 力 和 多 版 本 并 发 控制 的 事务 安全 。 在 MySQL 
从 3.23.34a 开始 包含 InnoDB 存储 引擎 。InnoDB 是 MySQL 上 第 一 个 提供 外 键 约束 的 表 引 擎 。 而 且 
InnoDB 对 事务 处 理 的 能 力 ， 也 是 MySQL 其 他 存储 引擎 所 无 法 与 之 比拟 的 。 下 面 介绍 InnoDB 存储 引 
擎 的 特点 及 其 优 缺 点 。 

InnoDB 存储 引擎 中 支持 自动 增长 列 AUTO_INCREMENT。 自 动 增长 列 的 值 不 能 为 空 ， 且 值 必须 
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唯一 。MySQL 中 规定 自 增 列 必须 为 主键 。 在 插入 值 时 ， 如 果 自 动 增长 列 不 输入 值 ， 则 插入 的 值 为 自动 
增长 后 的 值 ; 如 果 输 入 的 值 为 0 或 室 (NULL) ， 则 插入 的 值 也 为 自动 增长 后 的 值 ; 如 果 插 入 某 个 确定 
的 值 ， 且 该 值 在 前 面 没有 出 现 过 ， 则 可 以 直接 插入 。 

InnoDB 存储 引擎 中 支持 外 键 (FOREIGN KEY) 。 外 刍 所 在 的 表 为 子 表 ， 外 键 所 依赖 的 表 为 父 表 。 
父 表 中 被 子 表 外 键 关联 的 字段 必须 为 主键 。 当 删除 、 更 新 父 表 的 某 条 信息 时 ， 子 表 也 必须 有 相应 的 改 
变 .InnoDB 存储 引擎 中 ,创建 的 表 的 表 结 构 存 储 在 .frm 文 件 中 .数据 和 索引 存储 在 innodb_data_ home dir 
和 innodb_data_file_ path 表 空 间 中 。 

InnoDB 存储 引擎 的 优势 在 于 提供 了 和 良好 的 事务 管理 、 崩 溃 修 复 能 力 和 并 发 控制 。 缺 点 是 其 读 写 效 
率 稍 差 ， 占 用 的 数据 空间 相对 比较 大 。 

InnoDB 表 是 如 下 情况 的 理想 引擎 。 

(1) 更 新 密集 的 表 : InnoDB 存储 引擎 特别 适合 处 理 多 重 并 发 的 更 新 请 求 。 

(2) 事务 : InnoDB 存储 引擎 是 唯一 支持 事务 的 标准 MySQL 存储 引擎 ， 这 是 管理 敏感 数据 (如 
金融 信息 和 用 户 注册 信息 ) 的 必需 软件 。 

(3) 自动 灾难 恢复 : 与 其 他 存储 引擎 不 同 ，InnoDB 表 能 够 自动 从 灾难 中 恢复 。 虽 然 MyISAM 表 
能 在 灾难 后 修复 ， 但 其 过 程 要 长 得 多 。 

Oracle 的 InnoDB 存储 引擎 广泛 应 用 于 基于 MySQL 的 Web、 电 子 商务 、 金 融 系 统 、 健 康 护 理 以 
及 零售 应 用 。 因 为 InnoDB 可 提供 高 效 的 ACID 独立 性 (Atomicity) 、 一 致 性 〈Consistency) 、 隔 离 性 
(Isolation)、 持 久 性 (Durability) 兼容 事务 处 理 能 力 ， 以 及 独特 的 高 性 能 和 具有 可 扩展 性 的 构架 要 素 。 

另外 ，InnoDB 设计 用 于 事务 处 理应 用 ， 这 些 应 用 需要 处 理 崩 溃 恢 复 、 参 照 完 整 性 、 高 级 别 的 用 户 
并 发 数 ， 以 及 响应 时 间 超时 服务 水 平 。 在 MySQL 5.5 中 ， 最 显著 的 增强 性 能 是 将 InnoDB 作为 默认 的 
存储 引擎 。 在 MyISAM 以 及 其 他 表 类 型 依然 可 用 的 情况 下 , 用 户 无 须 更 改 配 置 , 就 可 构建 基于 InnoDB 
的 应 用 程序 。 





5.1.4 ”MylSAM 存储 引擎 


MyISAM 存储 引擎 是 MySQL 中 常见 的 存储 引擎 ， 曾 是 MySQL 的 默认 存储 引擎 。MyISAM 存储 引 
擎 是 基于 ISAM 存储 引擎 发 展 起 来 的 ， 它 解决 了 ISAM 的 很 多 不 足 。MyISAM 增加 了 很 多 有 用 的 扩展 。 


1. MyISAM 存储 引擎 的 文件 类 型 


MyISAM 存储 引擎 的 表 存储 成 3 个 文件 。 文 件 的 名 字 与 表 名 相同 , 扩展 名 包括 fm、MYD 和 MYT。 
(1) frm: 存储 表 的 结构 。 

(2) MYD: 存储 数据 ， 是 MYData 的 缩写 。 

(3) MYI: 存储 索引 ， 是 MYIndex 的 缩写 。 


2. MylSAM 存储 引擎 的 存储 格式 


基于 MyISAM 存储 引擎 的 表 支 持 3 种 不 同 的 存储 格式 ， 包 括 静 态 型 、 动 态 型 和 压缩 型 。 
1) MyISAM 静态 
如 果 所 有 表 列 的 大 小 都 是 静态 的 ( 即 不 使 用 xBLOB、xTEXT 或 VARCHAR 数据 类 型 ) ，MySQL 
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就 会 自动 使 用 静态 MyISAM 格式 。 使 用 这 种 类 型 的 表 性 能 非常 高 ， 因 为 在 维护 和 访问 以 预定 义 格 式 存 
储 的 数据 时 需要 很 低 的 开销 。 但 是 ， 这 项 优点 要 以 空间 为 代价 ， 因 为 每 列 都 需要 分 配给 该 列 最 大 空间 ， 
而 无 论 该 空间 是 否 真正 地 使 用 。 

2) MyISAM 动态 

如 果 有 表 列 (即使 只 有 一 列 ) 定 义 为 动态 的 (使 用 xBLOB、xTEXT 或 VARCHAR) ，MySQL 就 
会 自动 使 用 动态 格式 。 虽然 MyISAM 动态 表 占 用 的 空间 比 静 态 格式 所 占 空 间 少 但 空间 的 节省 带 来 了 
性 能 的 下 降 。 如 果 某 个 字段 的 内 容 发 生 改 变 ， 则 其 位 置 很 可 能 就 需要 移动 ， 这 会 导致 碎片 的 产生 。 随 
着 数据 集中 的 碎片 增加 ， 数 据 访问 性 能 就 会 相应 降低 。 这 个 问题 有 以 下 两 种 修复 方法 。 

(1) 尽 可 能 使 用 静态 数据 类 型 。 

(2) 经 常 使 用 OPTIMIZE TABLE 语句 ， 它 会 整理 表 的 碎片 ， 恢 复 由 于 表 更 新 和 删除 而 导致 的 空 
间 丢 失 。 

3) MyISAM 压缩 

有 时 会 创建 在 整个 应 用 程序 生命 周期 中 都 只 读 的 表 。 如 果 是 这 种 情况 ， 就 可 以 使 用 myisampack 工 
有 具 将 其 转换 为 MyISAM 压缩 表 来 减少 空间 .在 给 定 硬件 配置 下 (如 快速 的 处 理 器 和 低速 的 硬盘 驱动 器 )， 
性 能 的 提升 将 相当 显著 。 


3. MyISAM 存储 引擎 的 优 缺点 
MyISAM 存储 引擎 的 优势 在 于 占用 空间 小 ， 处 理 速度 快 ， 缺点 是 不 支持 事务 的 完整 性 和 并 发 性 。 





5.1.5 ” MEMORY 存储 引擎 


MEMORY 存储 引擎 是 MySQL 中 的 一 类 特殊 的 存储 引擎 。 其 使 用 存储 在 内 存 中 的 内 容 来 创建 表 ， 
而 且 所 有 数据 也 放 在 内 存 中 。 这 些 特 性 都 与 InnoDB 存储 引擎 、MyISAM 存储 引擎 不 同 。 下 面 将 对 
MEMORY 存储 引擎 的 文件 存储 形式 、 索 引 类 型 、 存 储 周 期 和 优 缺 点 等 进行 讲解 。 

1. MEMORY 存储 引擎 的 文件 存储 形式 


每 个 基于 MEMORY 存储 引擎 的 表 实际 对 应 一 个 磁盘 文件 。 该 文件 的 文件 名 与 表 名 相同 ， 类 型 为 
frm。 该 文件 中 只 存储 表 的 结构 ， 而 其 数据 文件 都 是 存储 在 内 存 中 。 这 样 有 利于 对 数据 的 快速 处 理 ， 提 
高 整个 表 的 处 理 效 率 。 值 得 注意 的 是 ， 服 务 器 需要 有 足够 的 内 存 来 维持 MEMORY 存储 引擎 的 表 的 使 
用 。 如 果 不 需要 使 用 了 ， 可 以 释放 这 些 内 容 ， 甚 至 可 以 删除 不 需要 的 表 。 


2. MEMORY 存储 引擎 的 索引 类 型 


MEMORY 存储 引擎 默认 使 用 哈 希 (HASH) 索引 ， 其 速度 要 比 使 用 B 树 (BTREE) 索引 快 。 如 
果 读 者 希望 使 用 B 树 索引 ， 可 以 在 创建 索引 时 选择 使 用 。 


3. MEMORY 存储 引擎 的 存储 周期 


MEMORY 存储 引擎 通常 很 少 用 到 。 因为 MEMORY 表 的 所 有 数据 是 存储 在 内 存 上 的 , 如 果 内 存 出 
现 异常 就 会 影响 到 数据 的 完整 性 。 如 果 重 启 机 器 或 者 关机 ， 表 中 的 所 有 数据 将 消失 。 因 此 ， 基 于 
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MEMORY 存储 引擎 的 表 生 命 周期 很 得 ， 一 般 都 是 一 次 性 的 。 
4. MEMORY 存储 引擎 的 优 缺 点 


MEMORY 表 的 大 小 是 受到 限制 的 。 表 的 大 小 主要 取决 于 两 个 参数 ， 分 别 是 max rows 和 
max_heap table size。 其 中 , max_rows 可 以 在 创建 表 时 指定 ; max_heap _table size 的 大 小 默认 为 16MB， 
可 以 按 需 要 进行 扩大 。 因 此 ， 其 存在 于 内 存 中 的 特性 ， 决 定 了 这 类 表 的 处 理 速度 非常 快 。 但 是 ， 其 数 
据 易 丢失 ， 生 命 周 期 短 。 

创建 MySQL MEMORY 存储 引擎 的 出 发 点 是 速度 。 为 得 到 最 快 的 响应 时 间 ， 采 用 的 逻辑 存储 介质 
是 系统 内 存 。 虽 然 在 内 存 中 存储 表 数据 确实 会 提高 性 能 ， 但 要 记 住 ， 当 mysqld 守护 进程 崩溃 时 ， 所 有 
的 MEMORY 数据 都 会 丢失 。 

MEMORY 表 不 支持 VARCHAR、BLOB 和 TEXT 数据 类 型 ， 因 为 这 种 表 类 型 按 固定 长 度 的 记录 
格式 存储 。 此 外 , 如 果 使 用 版 本 4.1.0 之 前 的 MySQL, 则 不 支持 自动 增加 列 (通过 AUTO_INCREMENT 
属性 ) 。 当 然 ， 要 记 住 MEMORY 表 只 用 于 特殊 的 范围 ， 不 会 用 于 长 期 存储 数据 。 基 于 其 这 个 缺陷 ， 
选择 MEMORY 存储 引擎 时 要 特别 小 心 。 

当 数 据 有 如 下 情况 时 ， 可 以 考虑 使 用 MEMORY 表 。 

(1) 和 暂时: 目标 数据 只 是 临时 需要 ， 在 其 生命 周期 中 必须 立即 可 用 。 

(2) 相对 无 关 : 存储 在 MEMORY 表 中 的 数据 如 果 突 然 丢 失 ， 不 会 对 应 用 服务 产生 实质 的 负面 影 
响 ， 而 且 不 会 对 数据 完整 性 有 长 期 影响 。 

如 果 使 用 MySQL 4.1 及 其 之 前 版 本 ，MEMORY 的 搜索 比 MyISAM 表 的 搜索 效率 要 低 ， 因 为 
MEMORY 表 只 支持 散 列 索引 ， 这 需要 使 用 整个 键 进行 搜索 。 但 是 ，4.1 之 后 的 版 本 同时 支持 散 列 索引 
和 B 树 索引 。B 树 索引 优 于 散 列 索引 的 是 ， 可 以 使 用 部 分 查询 和 通 配 查 询 ， 也 可 以 使 用 <、> 和 > 等 操 
作 符 方 便 数据 挖掘 。 

















5.1.6 ”如 何 选 择 存储 引擎 


每 种 存储 引擎 都 有 各 自 的 优势 ， 不 能 笼统 地 说 谁 比 谁 更 好 ， 只 有 适合 不 适合 。 下 面 根据 其 不 同 的 
特性 ， 给 出 选择 存储 引擎 的 建议 。 

(1) InnoDB 存储 引擎 : 用 于 事务 处 理应 用 程序 ， 具 有 众多 特性 ， 包 括 ACID 事务 支持 ， 支 持 外 
键 。 同 时 支持 崩溃 修复 能 力 和 并 发 控制 。 如 果 需 要 对 事务 的 完整 性 要 求 比较 高 ， 要 求实 现 并 发 控制 ， 
那 选择 InnoDB 存储 引擎 有 其 很 大 的 优势 。 如 果 需 要 频繁 地 进行 更 新 、 删 除 操作 的 数据 库 ， 也 可 以 选择 
InnoDB 存储 引擎 ， 因 为 该 类 存储 引擎 可 以 实现 事务 的 提交 (Commit) 和 回 滚 (Rollback) 。 

(2) MyISAM 存储 引擎 : 管理 非 事 务 表 ， 它 提供 高 速 存储 和 检索 ， 以 及 全 文 搜索 能 力 。MyYISAM 
存储 引擎 插入 数据 快 ， 空 间 和 内 存 使 用 比较 低 。 如 果 表 主要 是 用 于 插入 新 记录 和 读 出 记录 ， 那 么 选择 
MyISAM 存储 引擎 能 实现 处 理 的 高 效率 。 如 果 应 用 的 完整 性 、 并 发 性 要 求 很 低 ， 也 可 以 选择 MyISAM 
存储 引擎 。 

(3) MEMORY 存储 引擎 : MEMORY 存储 引擎 提供 “内 存 中 ”的 表 ，MEMORY 存储 引擎 的 所 
有 数据 都 在 内 存 中 ， 数 据 的 处 理 速度 快 ， 但 安全 性 不 高 。 如 果 需 要 很 快 的 读 写 速度 ， 对 数据 的 安全 性 
要 求 较 低 ， 可 以 选择 MEMORY 存储 引擎 。MEMORY 存储 引擎 对 表 的 大 小 有 要 求 ， 不 能 建 太 大 的 表 。 
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所 以 ， 这 类 数据 库 只 使 用 相对 较 小 的 数据 库 表 。 
以 上 存储 引擎 的 选择 建议 是 根据 不 同 存储 引擎 的 特点 提出 的 ， 并 不 是 绝对 的 。 实 际 应 用 中 还 需要 
根据 各 自 的 实际 情况 进行 分 析 。 


5.1.7 ”设置 数据 表 的 存储 引擎 


下 面 创建 db_database03 数据 库 文件 ， 在 数据 库 中 创建 3 个 数据 表 ， 并 分 别 为 其 设置 不 同 的 存储 引 

擎 。 以 此 来 诠释 这 3 种 不 同 存储 引擎 创建 的 数据 表 文 件 有 什么 区 别 。 
(1) 创建 tb_001 数据 表 ， 设 置 存储 引擎 为 MYISAM， 生 成 的 数据 表 文 件 如 图 5.4 所 示 ， 由 3 个 不 
同 后 级 的 文件 组 成 。 











| 
| 

本 Cm . Bo sr 5 
| 
| 





= dbopt 1K| 
perm 和 tb.001frm 9 
Dtb.o0LMmp ok 

Dtb.o01Mm 1k 


加 mn » 
本 


图 5.4 创建 也 001 数据 表 及 生成 的 数据 表 文 件 


(2) 创建 tb_002 数据 表 ， 设 置 存储 引擎 为 MEMORY， 生 成 的 数据 表 文 件 如 图 5.5 所 示 ， 只 有 一 
个 后 缀 为 frm 的 文件 。 




















EE 太 小 
Ddopt 1 
BS tb_001frm 9 K| 
口 tboolMyD ox| 
Dtb.00LMY 1 K| 














BS 也 -002frm 9 k| 





图 5.5 创建 也 002 数据 表 及 生成 的 数据 表 文 件 


(3) 创建 tb_003 数据 表 ， 设 置 存储 引擎 为 mnoDB， 生 成 的 数据 表 文 件 如 图 5.6 所 示 ， 同 样 也 由 
一 个 后 级 为 frm 的 文件 组 成 。 
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】 名 和 > 大 小 
ee 站 国法 党 
PARTMON definibon: oy B tb_001frm 9 KI 
jtb_001 MYD 0 Kg 
tb 00LMYI 1 
3 tb_002frm gk 
BG tb_003frm 9 Kl 





图 5.6 创建 tb_003 pepe 
5.2 ”MySQL 数据 类 型 


在 MySQL 数据 库 中 ， 每 一 条 数据 都 有 其 数据 类 型 。MySQL 支持 的 数据 类 型 主要 分 成 3 类 : 
数字 类 型 、 字 符 串 〈 字 符 ) 类 型 、 日 期 和 时 间 类 型 。 


5.2.1 数字 类 型 


MySQL 支持 所 有 的 ANSIISO SQL 92 数字 类 型 。 这些 类 型 包括 准确 数字 的 数据 类 型 (NUMERIC、 
DECIMAL、INTEGER 和 SMALLINT) ， 还 包括 近似 数字 的 数据 类 型 (FLOAT、REAL 和 DOUBLE 
PRECISION) 。 其 中 的 关键 词 INT 是 INTEGER 的 同义词 ， 关 键 词 DEC 是 DECIMAL 的 同义词 。 

数字 类 型 总 体 可 以 分 成 整 型 和 浮 点 型 两 类 ， 详 细 内 容 如 表 5.1 和 表 5.2 所 示 。 


表 5.1 整数 数据 类 型 























数据 类 型 取 值 范 说 明 单 位 
TINYINT 符号 值 ，-127~127 无 符号 值 : 0~255 最 小 的 整数 1 学 蕊 
BIT 符号 值 : -127~127 无 符号 值 : 0~255 最 小 的 整数 1 字 节 
BOOL 符号 值 ，-127~127 无 符号 值 : 0~255 最 小 的 整数 1 字 节 
符号 值 : -32 768~32 767 

S oy 无 符号 值 : 0-65 535 小 型 整数 2 字 节 
符号 值 : -8 388 608~8 388 607 

无 符号 值 ，0~16 777 215 中 型 整数 3 字 节 
符号 值 : - 2 147 683 648~2 147 683 647 要 二 

四 无 符号 值 ，0-4 294 967 295 标准 整数 4 字 节 
符号 值 : 

BIGINT —9 223 372 036 854 775 808~9 223 372 036 854 775 807 大 整数 8 字 节 
无 符号 值 : 0~18 446 744 073 709 551 615 
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表 5.2 浮 点 数据 类 型 
















取 值 范 说 有明 单 位 
FLOAT | 十 (-) 3.402 823 466E+38 单 精度 浮 点 数 | 8 或 4 字 节 
+ 〈-) 1.797 693 134 862 315 7E+308 
DOUBLE | 双 精 度 浮 点 数 | 8 字 节 


+ (-) 2.225 073 858 507 201 4E-308 
可 变 









一 般 整数 自 定义 长 度 





在 创建 表 时 ， 使 用 哪 种 数字 类 型 ， 应 遵循 以 下 原则 。 

(1) 选择 最 小 的 可 用 类 型 ， 如 果 值 永远 不 超过 127， 则 使 用 TINYINT 比 INT 强 。 
(2) 对 于 完全 都 是 数字 的 ， 可 以 选择 整数 类 型 。 

(3 ) 浮 点 类 型 用 于 可 能 具有 小 数 部 分 的 数 ， 如 货物 单价 、 网 上 购物 交付 金额 等 。 


5.2.2 ”字符 串 类 型 


字符 串 类 型 可 以 分 为 3 类 : 普通 的 文本 字符 串 类 型 (CHAR 和 VARCHAR) 、 可 变 类 型 (TEXT 和 
BLOB) 和 特殊 类 型 (SET 和 ENUM) 。 它 们 之 间 都 有 一 定 的 区 别 ， 取 值 范围 不 同 ， 应 用 的 地 方 也 不 同 。 

(1) 普通 的 文本 字符 串 类 型 ， 即 CHAR 和 VARCHAR 类 型 ，CHAR 列 的 长 度 被 固定 为 创建 表 所 
声明 的 长 度 ， 取 值 在 1~255 之 间 ; VARCHAR 列 的 值 是 变 长 的 字符 串 ， 取 值 和 CHAR 一样。 普通 的 文 
本 字符 串 类 型 的 介绍 如 表 5.3 所 示 。 


表 5.3 常规 字符 串 类 型 








类 型 取 值 范 说 明 
i 固定 长 度 为 M 的 字符 串 ， 其 中 M 的 取 值 范 围 为 0~255。national 关键 
har(M) 0-255 个 字符 字 指定 了 应 该 使 用 的 默认 字符 集 。binary 关键 字 指 定 了 数据 是 否 区 分 大 
小 写 (默认 是 区 分 大 小 写 的 ) 。 ASCII 关键 字 指定 了 在 该 列 中 使 用 latin1 
本 字符 集 。unicode 关键 字 指定 了 使 用 UCS 字符 集 
char 0-255 个 字符 “| char(M) 类 似 
[national] 
varchar(M) 0~255 个 字符 “| 长 度 可 变 ， 其 他 和 char(M) 类 似 
[binary] 








(2) 可 变 类 型 (TEXT 和 BLOB) 。 它 们 的 大 小 可 以 改变 ，TEXT 类 型 适合 存储 长 文本 ,而 BLOB 
类 型 适合 存储 二 进 制 数据 ， 支 持 任何 数据 ， 如 文本 、 声 音 和 图 像 等 。TEXT 和 BLOB 类 型 的 介绍 如 
表 5.4 所 示 。 
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表 5.4 TEXT 和 BLOB 类 型 


























类 型 最 大 长 度 〈 字 节 数 ) 说 明 
TINYBLOB 2^8-1 (255) 小 BLOB 字段 
TINYTEXT 2^8-1 (255) 小 TEXT 字段 
BLOB 2^16-1 (65 535) 常规 BLOB 字段 
TEXT 2^16-1 (65 535) 常规 TEXT 字段 
MEDIUMBLOB 2^24-1 (16 777 215) 中 型 BLOB 字段 
MEDIUMTEXT 2^24-1 (16 777 215) 中 型 TEXT 字段 
LONGBLOB 2^32-1 (4 294 967 295) 长 BLOB 字段 
LONGTEXT 2^32-1 (4 294 967 295) 长 TEXT 字段 


(3) 特殊 类 型 (SET 和 ENUM) 。 
特殊 类 型 (SET 和 ENUM) 的 介绍 如 表 5.5 所 示 。 


表 5.5 ENUM 和 SET 类 型 


类 型 


Enum ("valuel", "value2", ...) 





Set ("valuel", "value2", ...) 


AS 名 明 
创建 表 时 ， 使 用 字符 串 类 型 时 应 遵循 以 下 原则 。 
(1) 从 速度 方面 考虑 ， 要 选择 固定 的 列 ， 可 以 使 用 CHAR 类型。 
(2) 要 节省 空间 ， 使 用 动态 的 列 ， 可 以 使 用 VARCHAR 类 型 。 
(3) 要 将 列 中 的 内 容 限制 在 一 种 选择 ， 可 以 使 用 ENUM 类 型 。 
(4) 允许 在 一 个 列 中 有 多 于 一 个 的 条 目 ， 可 以 使 用 SET 类 型 。 
(5) 如 果 要 搜索 的 内 容 不 区 分 大 小 写 ， 可 以 使 用 TEXT 类 型 。 
(6) 如 果 要 搜索 的 内 容 区 分 大 小 写 ， 可 以 使 用 BLOB 类 型 。 


5.2.3 日 期 和 时 间 类 型 
日 期 和 时 间 类 型 包括 : DATETIME、DATE、TIMESTAMP、TIME 和 YEAR。 其 中 的 每 种 类 型 


都 有 其 取 值 的 范围 ， 如 赋予 它 一 个 不 合法 的 值 ， 将 会 被 “0” 代 蔡 。 日 期 和 时 间 类 型 的 介绍 如 表 5.6 
所 示 。 
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表 5.6 日 期 和 时 间 数 据 类 型 











类 型 取 值 范围 说 明 
DATE 1000-01-01 9999-12-31 日 期 , 格式 YYYY-MM-DD 
TIME -838:58:59 835:59:59 时 间 ， 格 式 HH:MM:SS 
DATETIME pO 日 期 和 时 间 ， 格 式 YYYY-MM-DD HH:MM:SS 


9999-12-31 23:59:59 
1970-01-01 00:00:00 
2037 年 的 某 个 时 间 
YEAR 1901-2155 





时 间 标 签 ， 在 处 理 报告 时 使 用 显示 格式 取决 于 M 
的 值 
年 份 可 指定 两 位 数字 和 四 位 数字 的 格式 











在 MySQL 中 ， 日 期 的 顺序 是 按照 标准 的 ANSI SQL 格式 进行 输出 的 。 





3.3 外 结 


本 章 对 MySQL 存储 引擎 和 数据 类 型 分 别 进行 了 详细 讲解 ， 并 通过 举例 说 明 ， 帮 助 读 者 更 好 地 理 
解 所 学 知识 的 用 法 。 在 阅读 本 章 时 ， 读 者 应 该 重点 掌握 什么 类 型 的 表 适 合 什么 类 型 的 存储 引擎 ， 同 
时 对 MySQL 中 的 数据 类 型 也 要 有 一 定 的 了 解 , 在 以 后 设计 数据 表 时 ， 能 够 合理 地 选择 所 使 用 的 数据 
类 型 。 








5.4 实践 与 练习 


1. 查询 MySQL 中 支持 的 存储 引擎 ， 并 且 以 友好 效果 进行 显示 。 (答案 位 置 : 光盘 \TMNsI\S\S.2) 
2. 查询 默认 的 存储 引擎 ， 并 且 以 友好 效果 进行 显示 。 (答案 位 置 : 光盘 \TMNsIS\S.3) 
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第 章 
操作 数据 表 


(a 视频 讲解 ，12 分 钟 ) 


在 对 MySQL 数据 表 进 行 操作 之 前 ， 必 须 首先 使 用 USE 语句 选择 数据 库 ， 才 可 
在 指定 的 数据 库 中 对 数据 表 进 行 操作 ， 如 创建 数据 表 、 修 改 表 结 构 、 数 据 表 更 名 或 
删除 数据 表 等 ; 否则 是 无 法 对 数据 表 进 行 操作 的 。 本 章 将 对 数据 表 的 操作 方法 进行 
详细 介绍 。 

通过 阅读 本 章 ， 读 者 可 以 : 
掌握 创建 数据 表 的 方法 
了 解 查看 数据 表 的 方法 
掌握 修改 数据 表 结 构 的 方法 
掌握 重 命名 、 复 制 和 删除 数据 表 的 方法 


于 于 至 至 
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6.1 创建 数据 表 


创建 数据 表 使 用 CREATE TABLE 语句 。 语 法 如 下 。 


CREATE [TEMPORARY] TABLE [IF NOT EXISTS] 数据 表 名 
[(create_definition,….)][table_options] [select_statement] 


CREATE TABLE 语句 的 参数 说 明 如 表 6.1 所 示 。 
表 6.1 CREATE TABLE 语句 的 参数 说 明 


关 键 字 说 有明 

TEMPORARY 如 果 使 用 该 关键 字 ， 表 示 创 建 一 个 临时 表 

IF NOT EXISTS 该 关键 字 用 于 避免 表 存在 时 MySQL 报告 的 错误 

create_definition 这 是 表 的 列 属性 部 分 。MySQL 要 求 在 创建 表 时 ， 表 要 至 少 包 含 一 列 

as 表 的 一 些 特性 参数 ， 其 中 大 多 数 选 项 涉及 的 是 表 数 据 如 何 存储 及 存储 在 何 处 ， 如 ENGINE 
选项 用 于 定义 表 的 存储 引擎 。 多 数 情况 下 ， 用 户 不 必 指 定 表 选 项 

select_statement SELECT 语句 描述 部 分 ， 用 它 可 以 快速 创建 表 


下 面 介 绍 列 属性 create_definition 部 分 ， 每 一 列 定义 的 具体 格式 如 下 。 


col_name type [NOT NULL | NULL] [DEFAULT default_value] [AUTO_INCREMENT] 
[PRIMARY KEY ] [reference_definition] 


属性 create_definition 的 参数 说 明 如 表 6.2 所 示 。 
表 6.2 属性 create_definition 的 参数 说 明 
参数 说 上 明 





col name 字段 名 





type 字段 类 型 


NOT NULL | NULL 使 用 NOT 工 


DEFAULT default_value 表示 默认 值 


指出 该 列 是 否 允 许 是 空 值 ， 系 统一 般 默认 允许 为 空 值 ， 所 以 当 不 允许 为 空 值 时 ， 必 须 





AUTO INCREMENT 表示 是 否 是 自动 编号 ， 每 个 表 只 能 有 一 个 AUTO_INCREMENT 列 ， 并 且 必 须 被 索引 





表示 是 否 为 主键 。 一 个 表 只 能 有 一 个 PRIMARY KEY。 如 表 中 没有 一 个 PRIMARY 


PRIMARY KEY KEY， 而 某 些 应 用 程序 需要 PRIMARY KEY，MySQL 将 返回 第 一 个 没有 任何 NULL 


列 的 UNIQUE 键 ， 作 为 PRIMARY KEY 








reference_definition 为 字段 添加 注释 

















以 上 是 创建 一 个 数据 表 的 一 些 基础 知识 ， 它 看 起 来 十 分 复杂 ， 但 在 实际 的 应 
式 创 建 数据 表 即 可 ， 具 体格 式 如 下 。 


器 





ph 使 用 最 基本 的 格 
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CREATE TABLE 数据 表 名 ( 列 名 1 属性 , 列 名 2 属性 …); 


例 6.1 使 用 CREATE TABLE 语句 在 MySQL 数据 库 db_admin 中 创建 一 个 名 为 tb_admin 的 数据 
表 ， 该 表 包 括 id、user、password 和 createtime 等 字段 ， 具体 代码 如 下 。( 实 例 位 置 光盘 \TMNsN6\6.1) 


USE db_admin; 

CREATE TABLE tb_admin( 
id int auto_increment primary key, 
user varchar(30) not null, 
password varchar(30) not null, 
createtime datetime); 


执行 结果 如 图 6.1 所 示 。 
db_adnin 
reate table th. 


ia int auto_i ment primary ksy 
-> user varcharC30) null, 


ord varchar(39》 not null, 
ime dateti 


affected CG@.86 sec》 


Pysal> L 国 @ 数据 表 创建 成 功 


图 6.1 创建 MySQL 数据 表 





4 
说 明 
在 完成 本 实例 前 ， 如 果 不 存 在 名 称 为 db_admin 的 数据 库 ， 那 么 需要 先 创建 该 数据 库 ， 创 建 数 
据 库 db admin 的 具体 代码 如 下 。 


CREATE DATABASE db_ admin: 


6.2 ”查看 表 结 构 


对 于 一 个 创建 成 功 的 数据 表 ， 可 以 使 用 SHOW COLUMNS 语句 或 DESCRIBE 语句 查看 指定 数据 
表 的 表 结 构 。 下 面 分 别 对 这 两 个 语句 进行 介绍 。 


6.2.1 使 用 SHOW COLUMNS 语句 查看 
在 MySQL 中 , 使 用 SHOW COLUMNS 语句 可 以 查看 表 结 构 , SHOW COLUMNS 语句 的 基本 语法 


格式 如 下 。 
SHOW [FULL] COLUMNS “FROM 数据 表 名 [FROM 数据 库 名 ]: 
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或 
SHOW [FULL] COLUMNS “FROM 数据 表 名 .数据 库 名 ; 


例 6.2 使 用 SHOW COLUMNS 语句 查看 数据 表 tb_admin 的 表 结 构 , 具体 代码 如 下 。( 实 例 位 置 : 
光盘 \TMsI6\6.2) 


SHOW COLUMNS FROM tb_admin FROM db_admin; 
执行 效果 如 图 6.2 所 示 。 


Inysql> SHOW COLUMNS FROM tb_admin FROM db_adnin; 和 be 
+ + + @ 查看 数据 表 结 构 
; Field 1 Null 1 Key ; Default 上 Extra | 
+ + 
1 NO 1 PRI ; NULL 1 auto_increment 上 
】 NO ; NULL 
1 NO ; NULL 
! YES ! ; NULL 


s in Set 《0.17 sec) 





图 62 查看 表 结构 
6.2.2 ”使 用 DESCRIBE 语句 查看 


在 MySQL 中 ， 还 可 以 使 用 DESCRIBE 语句 查看 数据 表 结 构 。DESCRIBE 语句 的 基本 语法 格式 
如 下 。 


DESCRIBE 数据 表 名 ; 

其 中 ，DESCRIBE 可 以 简写 成 DESC。 在 查看 表 结构 时 ， 也 可 以 只 列 出 某 一 列 的 信息 。 其 语法 格 
式 如 下 。 

DESCRIBE 数据 表 名 列 名 ; 

例 6.3 使 用 DESCRIBE 语句 的 简写 形式 查看 数据 表 tb_admin 中 的 某 一 列 信息 。 【实例 位 置 : 光 
盘 \TMNsI\6\6.3) 

(1) 编写 SQL 语句 ， 选 择 要 查看 数据 表 所 在 的 数据 库 ， 有 具体 代码 如 下 。 

USE db_admin; 

(2) 应 用 简写 的 DESC 命令 查看 数据 表 tb_admin 中 的 user 字段 的 信息 ， 具 体 代 码 如 下 。 

DESC tb_admin user'; 


执行 结果 如 图 6.3 所 示 。 
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sql> USE db_adnin; 
ee a @ 选择 数据 表 所 在 的 数据 库 


@ 使 用 简写 的 DESC 查看 数据 表 某 一 列 的 人 


! Field ! Type 1 Null ! Key ; Default ! Extra \ 


mysql> DESC tb_admin user; 
+ + + 


图 6.3 查看 表 的 某 一 列 信息 


6.3 修改 表 结 构 





目 数据 表 结 构 


修改 表 结构 使 用 ALTER TABLE 语句 。 修 改 表 结 构 指 增加 或 者 删除 字段 、 修 改 字段 名 称 或 者 字段 


类 型 、 设 置 取消 主键 外 键 、 设 置 取消 索引 以 及 修改 表 的 注释 等 ， 语 法 如 下 。 
ALTER [IGNORE] TABLE 数据 表 名 alter_specl[,alter_spec]...| table_options 


参数 说 明 如 下 。 


(1) [IGNORE]: 可 选项 ， 表 示 如 果 出 现 重复 关键 的 行 ， 则 只 执行 一 行 ， 其 他 重复 的 行 被 删除 。 





(2) 数据 表 名 : 用 于 指定 要 修改 的 数据 表 的 名 称 。 
(3) alter_ spec 子 句 : 用 于 定义 要 修改 的 内 容 ， 其 语法 格式 如 下 。 


ADD [COLUMN] create_definition [FIRST | AFTER column_name ] 
| ADD INDEX [index_name] (index_col_name,...) 

| ADD PRIMARY KEY (index_col_name,...) 

| ADD UNIQUE [index_name] (index_col_name,...) 

| ALTER [COLUMN] col_name {SET DEFAULT literal | DROP DEFAULT} 
| CHANGE [COLUMN] old_col_name create_definition 

| MODIFY [COLUMN] create_definition 

| DROP [COLUMN] col_name 

| DROP PRIMARY KEY 

| DROP INDEX index_name 

| RENAME [AS] new_tbl_name 


上 面 的 语法 中 ， 各 参数 说 明 如 下 。 


/添加 新 字段 
/添加 索引 名 称 
/添加 主键 名 称 
/添加 唯一 索引 
/修改 字段 默认 值 
/修改 字段 名 /类 型 
/修改 子 句 定义 字段 
/删除 字段 名 称 
/删除 主键 名 称 

1/ 删除 索引 名 称 

// 更 改 表 名 


GO create_definition: 用 于 定义 列 的 数据 类 型 和 属性 , 与 6.1 节 CREATE TABLE 语句 中 的 语法 相同 。 
@ [FIRST | AFTER column name ]: 用 于 指定 位 于 哪个 字段 的 前 面 或 者 后 面 ， 当 使 用 FIRST 关键 
字 时 ， 表 示 位 于 指定 字段 的 前 面 ; 使 用 AFTER 关键 字 时 ， 表 示 位 于 指定 字段 的 后 面 。 其 中 的 


column_ name 表示 字段 名 。 
@ [index_name]: 可 选项 ， 用 于 指定 索引 名 。 
@ (index_col name,…): 用 于 指定 索引 列 名 。 
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回 {SET DEFAULT literal | DROP DEFAULT} 子 句 : 为 字段 设置 或 者 删除 默认 值 。 其 中 literal 参数 
为 要 设置 的 默认 值 。 

@ old_col name: 用 于 指定 要 修改 的 字段 名 。 

@@ new _tbl name: 用 于 指定 新 的 表 名 。 

(4) table_options: 用 于 指定 表 的 一 些 特性 参数 ， 其 中 大 多 数 选 项 涉及 的 是 表 数 据 如 何 存储 及 存储 
在 何 处 ， 如 ENGINE 选项 用 于 定义 表 的 存储 引擎 。 多 数 情况 下 ， 用 户 不 必 指 定 表 选项 。 

人 涪 明 
ALTER TABLE 语句 允许 指定 多 个 动作 ， 其 动作 间 使 用 过 号 分 隔 ， 每 个 动作 表示 对 表 的 一 个 








6.3.1 添加 新 字段 及 修改 字段 定 


在 MySQL 的 ALTER TABLE 语句 中 ， 可 以 通过 使 用 ADD [COLUMN] create_definition [FIRST | 
AFTER column_name ] 子 句 来 添加 新 字段 ; 使 用 MODIFY [COLUMN] create_definition 子 句 可 以 修改 已 
定义 字段 的 定义 。 下 面 将 通过 一 个 具体 实例 演示 如 何 为 一 个 已 有 表 添 加 新 字段 ， 并 修改 已 有 字段 的 字 
段 定义 。 

例 6.4 添加 一 个 新 的 字段 email， 类 型 为 varchar(50)，not null， 将 字段 user 的 类 型 由 varchar(30) 
改 为 varchar(40)。《〈 实 例 位 置 ， 光盘 \TMNsN\6\6.4) 

(1) 选择 数据 库 db_admin， 具 体 代码 如 下 。 

USE db_admin; 

(2) 编写 SQL 语句 ， 实 现 向 数据 表 tb_admin 中 添加 一 个 新 字段 ， 并 且 修 改 字 段 user 的 类 型 ， 具 
体 代 码 如 下 。 


ALTER TABLEtb_admin ADD email varchar(50) not null , 
modify user varchar(40); 


在 命令 行 模式 下 的 运行 情况 如 图 6.4 所 示 。 


SE dhb_admin: @ 选择 数据 表 所 在 的 数据 库 


changed 


sql> ALTER TABLE tb_admin ADD email uarchar(59》 not null . 


> modify user va 
IlQuery OK. 1 row aff 
s: 1 Duplica 


@ 添加 新 字段 并 修改 字段 user 的 字段 类 型 











6.4 ”添加 新 字段 、 修 改 字段 类 型 
(3) 通过 DESC 命令 查看 数据 tb_user 的 表 结 构 ， 以 查看 表 结 构 是 否 成 功 修改 ， 具 体 代码 如 下 。 
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DESC tb_admin; 
执行 效果 如 图 6.5 所 示 。 


mysql> DESC thb_admin 


int C11> 

varchar C40. 

varcharC30> 

datet ime ES NULL 
varchar (50> NULL 


Net (‘0.01 sec)》 @ 将 原 字段 类 型 varchar(30) 修 改 为 varchar(40) 
@ 新 添加 的 字段 





图 6.5 修改 后 tb_user 的 表 结构 


说 明 
通过 ALTER 语句 修改 表 列 ， 其 前 提 是 必须 将 表 中 数据 全 部 删除 ， 然 后 才 可 以 修改 表 列 。 


6.3.2 ”修改 字段 名 


在 MySQL 的 ALTER TABLE 语句 中 ， 使 用 CHANGE [COLUMN] old_col name create_definition 
子 句 可 以 修改 字段 名 或 者 字段 类 型 。 下 面 将 通过 一 个 具体 实例 演示 如 何 修改 字段 名 。 

例 6.5 将 数据 表 tb_userNewl 的 字段 名 user 修改 为 usermname， 有 具体 代码 如 下 。《【〈 实 例 位 置 ， 光 盘 \ 
TMNsN\6\6.5) 


ALTER TABLE db_admin.tb_usernew1 
CHANGE COLUMN user username VARCHAR(30) NULL DEFAULT NULL :; 


执行 效果 如 图 6.6 所 示 。 


Inysql> ALTER TABLE db_adnin.th_u newl 
-> CHANGE COLUMN user username UARCHARC3@> NULL DEFAULT NULL ; 


日 rows affe dB.39 > 


日 Duplicat @ Warnin 





图 6.6 修改 字段 名 
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6.3.3 ”删除 字段 


在 MySQL 的 ALTER TABLE 中 , 使 用 DROP [COLUMN] col name 子 句 可 以 删除 指定 字段 。 下 面 
将 通过 一 个 具体 实例 演示 如 何 删除 字段 。 
例 6.6 将 数据 库 db_ admin 中 的 数据 表 tb_userNewl 更 名 为 tb_userOld。 (实例 位 置 ， 光盘 \ 
TMN\sN\6\6.6) 
(1) 选择 数据 库 db_admin， 具 体 代 码 如 下 。 
USE db_admin; 
(2) 编写 SQL 语句 ， 实 现 将 数据 表 tb_admin 中 的 字段 email 删除 ， 具 体 代 码 如 下 。 
ALTER TABLE tb_admin DROP email; 
在 命令 行 模式 下 的 运行 情况 如 图 6.7 所 示 。 








图 6.7 删除 字段 


6.3.4 修改 表 名 


在 MySQL 的 ALTER TABLE 中 ， 使 用 RENAME [AS] new_tbl name 子 句 可 以 修改 表 名 。 下 面 将 
通过 一 个 具体 实例 演示 如 何 修改 表 名 。 
例 6.7 将 数据 库 db admin 中 的 数据 表 tb_userNewl 更 名 为 tb_userOld。 【实例 位 置 ， 光 盘 \ 
TMNsN\6\6.7) 
(1) 选择 数据 库 db_admin， 具 体 代 码 如 下 。 
USE db_admin; 
(2) 编写 SQL 语句 ， 实 现 将 数据 表 tb_userNew1l 更 名 为 tb_userOld， 有 具体 代码 如 下 。 
ALTER TABLE tb_usernew1 RENAME AS tb_userOld; 
在 命令 行 模式 下 的 运行 情况 如 图 6.8 所 示 。 


usernewl RENAME AS th_userOld; 





图 6.8 修改 表 名 
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6.4 重 命 名 表 


在 MySQL 中 ， 重 命名 数据 表 可 以 使 用 RENAME TABLE 语句 来 实现 。RENAME TABLE 语句 的 
基本 语法 格式 如 下 。 


RENAME TABLE 数据 表 名 1 To 数据 表 名 2 


oY 、 
和 说明 
该 语句 可 以 同时 对 多 个 数据 表 进 行 重 命名 ， 多 个 表 之 间 以 过 号 “,” 分 隔 。 
例 6.8 对 数据 表 tb_admin 进行 重 命名 , 更 名 后 的 数据 表 为 tb_user。( 实 例 位 置 : 光盘 \TMNshN6\6.8) 
(1) 使 用 RENAME 语句 将 数据 表 tb_admin 重 命名 为 tb_user， 有 具体 代码 如 下 。 
RENAME TABLE tb_admin TO tb_user; 
(2) 重 命名 后 ， 应 用 DESC 语句 查看 数据 表 tb_user 的 表 结 构 ， 具 体 代码 如 下 。 
DESC tb_user; 
执行 效果 如 图 6.9 所 示 。 


USE db_adnin; 
changed 
ql> RENAME TABLE tb_admin TO thb_u 
Query OK,. @ rows affect 《0@.86 sec》 


nysql> DESC th_user; 
| 1 Type 
mene AS 8 
! NO 
30> 1 NO 
38> 上 NO 
1 


nysql> 





创建 表 的 CREATE TABLE 命令 还 有 另外 一 种 语法 结构 ， 在 一 张 已 经 存在 的 数据 表 的 基础 上 创建 
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- 份 该 表 的 备份 ， 也 就 是 复制 表 。 这 种 用 法 的 语法 格式 如 下 。 


CREATE TABLE [IF NOT EXISTS] 数据 表 名 
{LIKE 源 数据 表 名 | (LIKE 源 数 据 表 名 )} 


参数 说 明 如 下 。 
(1) [FE NOT EXISTS]: 可 选项 ， 如 果 使 用 该 子 句 ， 表 示 当 要 创建 的 数据 表 名 不 存在 时 ， 才 会 创 
建 。 如 果 不 使 用 该 子 句 ， 当 要 创建 的 数据 表 名 存在 时 ， 将 出 现 错误 。 
(2) 数据 表 名 : 表示 新 创建 的 数据 表 的 名 ， 该 数据 表 名 必须 是 在 当前 数据 库 中 不 存在 的 表 名 。 
(3) {LIKE 源 数据 表 名 | (LIKE 源 数据 表 名 )}: 必 选 项 ， 用 于 指定 依照 哪个 数据 表 来 创建 新 表 ， 
也 就 是 要 为 哪个 数据 表 创建 副本 。 
和 说明 
使 用 该 语法 复制 数据 表 时 ,将 创建 一 个 与 源 数 据 表 相同 结构 的 新 表 ， 该 数据 表 的 列 名 、 数 据 类 
型 空 指定 和 索引 都 将 被 复制 ， 但 是 表 的 内 容 是 不 会 复制 的 。 因 此 ， 新 创建 的 表 是 一 张 空 表 。 如 果 想 
要 复制 表 中 的 内 容 ， 可 以 通过 使 用 AS( 查 询 表 达 式 ) 子 句 来 实现 。 


例 6.9 在 数据 库 db_admin 中 创建 一 份 数据 表 tb_user 的 备份 tb_userNew。 (实例 位 置 ， 光盘 \ 
TMNsIN6\6.9) 

(1) 选择 数据 表 所 在 的 数据 库 db_admin， 有 具体 代码 如 下 。 

USE db_admin; 

(2) 创建 一 份 数据 表 tb_user 的 备份 tb_userNew， 有 具体 代码 如 下 。 


CREATE TABLE tb_userNew 
LIKE tb_user; 


执行 效果 如 图 6.10 所 示 。 





图 6.10 创建 一 份 数 据 表 tb_user 的 备份 tb_userNew 
(3) 查看 数据 表 tb_user 和 tb_userNew 的 表 结 构 ， 具 体 代码 如 下 。 


DESC tb_user; 
DESC tb_userNew; 


执行 结果 如 图 6.11 所 示 。 


A 
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Ne EE。 @ 查看 源 数据 表 的 表 结构 


Null Key Default Extra 
PRI NULL 


! ! auto_increment 
! NULL : 


+ + 


! varcharC38> 
datet ine 


1 NULL 
NULL 


t 0.82 


Type ! Ke Default 


NULL auto_increment 
NULL 
1 NULL 
NULL 
+ 
set 《0.81 





图 6.11 查看 数据 表 tb_user 和 tb_userNew 的 表 结 构 
从 图 6.11 中 可 以 看 出 ， 数 据 表 tb_user 和 tb_userNew 表 结 构 是 一 样 的 
(4) 分 别 查看 数据 表 tb_user 和 tb_userNew 的 内 容 ， 具 体 代码 如 下 。 


SELECT * FROM tb_user; 
SELECT * FROM tb_userNew; 


执行 效果 如 图 6.12 所 示 。 
> SELECT * FROM th_user; 
tuser 上 assword createtime 


14-89-11 工 


LECT * FROM th_userNew; 
et (08.80 sec) 





图 6.12 查看 数据 表 了 b_user 和 tb_userNew 的 内 容 
从 图 6.12 中 可 以 看 出 ， 在 复制 表 时 并 没有 复制 表 中 的 数据 。 
(5) 如 果 在 复制 数据 表 时 ， 想 要 同时 复制 其 中 的 内 容 ， 那 么 需要 使 用 下 面 的 代码 来 实现 。 


CREATE TABLE tb_userNew1 
AS SELECT * FROM tb_user; 





执行 结果 如 图 6.13 所 示 。 
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Inysql> CREATE TABLE tb_userNewdl 
-> AS SELECT x FROM th _user; 
lQuery OK. 1 row affected (@.35 sec)» 


IRecords: 1 Duplicates: 8@ Varnings: @ 


Inysql> 。 





图 6.13 复制 数据 表 同时 复制 其 中 的 数据 
(6) 查看 数据 表 tb_userNewl 中 的 数据 ， 具 体 代 码 如 下 。 
SELECT * FROM tb_userNew1; 
执行 效果 如 图 6.14 所 示 。 


1 


Inysql> 。 





图 6.14 查看 新 复制 的 数据 表 tb_userNewl 的 数据 
从 图 6.14 中 可 以 看 出 ， 在 复制 表 的 同时 也 复制 了 表 中 的 数据 。 


6.6 删 除 表 


删除 数据 表 的 操作 很 简单 ， 同 删除 数据 库 的 操作 类 似 ， 使 用 DROP TABLE 语句 即 可 实现 。DROP 
TABLE 语句 的 基本 语法 格式 如 下 。 

DROP TABLE [IF EXISTS] 数据 表 名 ; 

参数 说 明 如 下 。 

(1) [FEXISTS]: 可 选项 ,用 于 在 删除 表 前 先 判断 是 否 存在 要 删除 的 表 ， 只 有 存在 时 ， 才 执行 删 
除 操作 ， 这 样 可 以 避免 要 删除 的 表 不 存在 时 出 现 错误 信息 。 

(2) 数据 表 名 : 用 于 指定 要 删除 的 数据 表 名 ， 可 以 同时 删除 多 张 数据 表 ， 多 个 数据 表 名 之 间 用 英 
文 半角 的 逗号 “,” 分 隔 。 

例 6.10 ”删除 数据 表 tb_user。《〈 实 例 位 置 ; 光盘 \TMsI6\6.10) 

(1) 选择 数据 表 所 在 的 数据 库 db_admin， 具 体 代 码 如 下 。 

USE db_admin; 

(2) 应 用 DROP TABLE 语句 删除 数据 表 tb_user， 具 体 代 码 如 下 。 
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DROP TABLE: 


执行 效果 如 图 6.15 所 示 。 


mysql> drop table th_user; 
Query OK, @ rows affected (0.00 secy) 


@ 删 除数 据 表 tb_user 


nysql> 
四 成 功 删除 数据 表 tb_user 


图 6.15 ”删除 数据 表 





息 6 注 意 
有 

删除 数据 表 的 操作 应 该 谨慎 使 用 。 一 旦 删除 了 数据 表 ， 那么 表 中 的 数据 将 会 全 部 清除 , 没有 备 
份 则 无 法 恢复 。 


在 删除 数据 表 的 过 程 中 ， 删 除 一 个 不 存在 的 表 将 会 产生 错误 ， 如 果 在 删除 语句 中 加 入 正 EXISTS 
关键 字 就 不 会 出 错 了 ， 格 式 如 下 。 
DROP TABLE IF EXISTS 数据 表 名 ; 





6.7 小 结 


: 章 主要 介绍 了 如 何 创建 数据 表 、 查 看 表 结 构 、 修 改 表 结 构 、 重 命名 表 、 复 制 表 和 删除 表 等 内 容 。 
其 中 ， 创 建 和 修改 表 这 两 部 分 内 容 比较 重要 ， 需 要 不 断 的 练习 才 会 对 这 两 部 分 了 解 得 更 加 透彻 。 而 且 ， 
这 两 部 分 很 容易 出 现 语法 错误 ， 必 须 在 练习 中 掌握 正确 的 语法 规则 。 创 建 表 和 修改 表 后 一 定 要 查看 表 
的 结构 ， 这 样 可 以 确认 操作 是 否 正确 。 删 除 表 时 一 定 要 特别 小 心 ， 因 为 删除 表 的 同时 会 删除 表 中 的 所 
有 数据 。 


6.8 实践 与 练习 


1. 编写 SQL 语句 ， 实 现 查看 数据 表 tb_admin 的 表 结 构 。 (答案 位 置 : 光盘 \TMsI6\6.11) 

2. 编写 SQL 语句 ， 实 现 为 数据 表 tb_userNew 的 user 字段 设置 默认 值 为 mr。 (答案 位 置 ， 光盘 \ 
TM™MN\sI\6\6.12) 

3. 编写 SQL 语句 ， 实 现 当 数据 表 tb_user 存在 的 情况 下 删除 该 数据 表 。 (答案 位 置 ， 光盘 \ 
TM™MN\sI\6\6.13) 
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第 7 章 MySQL 基础 

第 8 章 表 数 据 的 增 、 删 、 改 操作 
第 9 章 数据 查询 

第 10 章 常用 困 数 

第 11 章 索引 

第 12 章 视图 


各 于 于 于 于 至 


本 篇 介绍 MySQL 基础 ， 表 数据 的 增 、 删 、 改 操作 ， 数 据 查 询 ， 常 用 函数 、 索 
引 、 视 图 的 使 用 等 。 学 习 完 这 一 部 分 , 读者 能 够 了 解 和 就 乱 My5QL 及 其 常用 函数 ， 
使 用 SQL 操作 MySQL 数据 库 中 的 视图 ， 党 担 SQL 查询 、 子 查询 、 谈 套 查 询 、 连 接 
查询 的 用 法 等 。 


第 后 
时 

MySQL 基础 

(名 视频 讲解 ，24 分 钟 ) 


同 其 他 语言 一 样 ，My5QL 教 据 库 也 有 自己 的 运算 符 和 流程 控制 语句 。 本 章 将 
对 My5QL 的 运算 符 和 流程 控制 语句 进行 详细 介绍 。 

通过 阅读 本 章 ， 读 者 可 以 : 

掌握 MySQL 的 运算 符 

WV 掌握 MySQL 的 流程 控制 语句 


第 7 章 MySQL 基础 


7.1.1 算术 运算 符 


算术 运算 符 是 MySQL 中 最 常用 的 一 类 运算 符 。MySQL 支持 的 算术 运算 符 包括 加 、 减 、 乘 、 除 、 
求 余 。 如 表 7.1 所 示 为 算术 运算 符 的 符号 、 作 用 、 表 达 式 的 形式 。 


























表 7.1 算术 运算 符 
符 ”号 作 用 

加 法 运算 

机 减法 运算 

本 乘法 运算 

/ 除法 运算 

% 求 余 运 算 

DIV 除法 运算 ， 返 回 商 。 同 “/” 
MOD 算 ， 返 回 余数 。 同 “%” 








(说明 
加 (+)、 减 (-) 和 乘 (*) 可 以 同时 运算 多 个 操作 数 。 除 号 (/) 和 求 余 运算 符 (% ) 也 可 以 
同时 计算 多 个 操作 数 ， 但 是 这 两 个 符号 计算 多 个 操作 数 不 太 好 。DIV 和 MOD 这 两 个 运算 符 只 有 两 
个 参数 。 进 行 除法 和 求 余 的 运算 时 ， 如 果 x2 参数 是 0 时 ， 计 算 结 果 将 是 空 值 (NULL ) 。 


例 7.1 使 用 算术 运算 符 对 数据 表 tb_bookl 中 的 row 字段 值 进行 加 、 减 、 乘 、 除 运算 ， 计 算 结 果 
如 图 7.1 所 示 。【〔 实 例 位 置 ， 光盘 \TMsN7\7.1) 


row from th_hookl; 





图 7.1 使 用 算术 运算 符 计算 数据 
结果 输出 了 row 字段 的 原 值 ， 以 及 执行 算术 运算 符 后 得 到 的 值 。 
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7.1.2 ”比较 运算 符 


比较 运算 符 是 查询 数据 时 最 常用 的 一 类 运算 符 。SELECT 语句 中 的 条 件 语 句 经 常 要 使 用 比较 运算 
符 。 通 过 这 些 比较 运算 符 ， 可 以 判断 表 中 的 哪些 记录 是 符合 条 件 的 。 比 较 运 算 符 的 符号 、 名 称 和 应 用 
示例 如 表 7.2 所 示 。 
表 7.2 ”比较 运算 符 
运算 符 | 名 称 示 例 


























示 例 
= 等 于 id=5 idIS NOT NULL 
> 雪 平 id>5 id BETWEEN!]1 AND 15 
< 小 于 id<5 id IN (3.4.5) 
>= 大 于 等 于 id=>5 name NOT IN (shi.li) 
zx 小 于 等 于 id<=5 name LIKE (shioo) 
!= 或 二 不 等 于 id!=5 NOT LIKE 四 name NOT LIKE (shio%) 
IS NULL na idis null 常规 表达 式 name 正则 表达 式 
下 面 对 几 种 较 常用 的 比较 运算 符 进行 详解 。 


运算 符 “=” 
=” 用 来 判断 数字 、 字 符 串 和 表达 式 等 是 否 相等 。 如 果 相等 ， 返 回 1， 否 则 返回 0。 
Ne 
说 明 
在 运用 运算 符 “=” 判 断 两 个 字符 是 否 相同 时 ， 数 据 库 系统 都 是 根据 字符 的 ASCII 码 进行 判断 


的 。 如 果 ASCII 码 相 等 ， 则 表示 这 两 个 字符 相同 。 如 果 ASCII 码 不 相等 ， 则 表示 两 个 字符 不 同 。 
切记 空 值 (NULL ) 不 能 使 用 “= ”来 判断 


例 7.2 运用 运算 符 “=” 查 询 出 id 等 于 27 的 记录 ， 查 询 结 果 如 图 7.2 所 示 。 (实例 位 置 ， 光盘 
\TMNsIN7\7.2) 





图 7.2 使 用 “=” 查 询 记录 
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从 结果 中 可 以 看 出 ，id 等 于 27 的 记录 返 

2、 运 算 符 “<>” 和 “=” 

“<>” 和 “!=” 用 来 判断 数字 、 字 符 串 、 表 达 式 等 是 否 不 相等 。 如 果 不 相等 ， 则 返回 1; 否则 ， 返 
可 0。 这 两 个 符号 也 不 能 用 来 判断 空 值 (NULL) 。 

例 7.3 运用 运算 符 “ 一 ”和 “!=” 判 断 数 据 表 tb book 中 的 row 字段 值 是 否 等 于 1、41 或 24。 运 
算 结果 如 图 7.3 所 示 。 《实例 位 置 : 光盘 \TMsIN7\7.3) 


Ee] 





值 为 1，id 不 等 于 27 的 记录 ， 返 回 值 则 为 0。 






































m th_hook where id=275 





图 7.3 使 用 运算 符 “ 一 ”和 “!=” 判 断 数据 
结果 显示 返回 值 都 为 1， 这 表示 记录 中 的 row 字段 值 不 等 于 1、41、24。 
3. 运算 符 “>” 
“>” 用 来 判断 左边 的 操作 数 是 否 大 于 右边 的 操作 数 。 如 果 大 于 ， 返 回 1; 否则 ， 返 回 0。 同 样 空 
值 (NULL) 不 能 使 用 “>” 来 判断 。 
例 7.4 使 用 运算 符 “>” 来 判断 数据 表 tb_book 中 的 row 字段 值 是 否 大 于 90， 是 则 返回 1， 否 则 
返回 0， 空 值 返回 NULL。 运 算 结 果 如 图 7.4 所 示 。 (实例 位 置 ， 光盘 \TMNsIN7\7.4) 





图 7.4 使 用 运算 符 “>” 查 询 数据 


/ 
说 明 

运算 符 “<”、 运 算 符 “<=” 和 运算 符 “>=” 都 与 运算 符 “>” 如 出 一 加 ， 其 使 用 方法 基本 相 
同 ， 这 里 不 再 黄 述 。 


4. 运算 符 IS NULL 


IS NULL 用 来 判断 操作 数 是 否 为 空 值 (NULL) 。 操 作 数 为 NULL 时 ,结果 返回 1; 否则 ,返回 0。 
IS NOT NULL 刚好 与 IS NULL 相反 。 
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例 7.5 用 运算 符 IS NULL 来 判断 数据 表 tb_book 中 的 row 字段 值 是 否 为 空 值 ， 查 询 结果 如 图 7.5 
所 示 。 (实例 位 置 ， 光盘 \TMNsIN7\7.S) 





图 7.5 使 用 运算 符 IS NULL 来 判断 字段 值 是 否 为 空 





py 
“=” “<>” “I=” “>” “>=” “<” “<=” 等 运算 符 都 不 能 用 来 判断 空 值 (NULL ) 。 一 
使 用 ,结果 将 返回 NULL。 如 果 要 判断 一 个 值 是 否 为 空 值 ， 可 以 使 用 “<=>”、IS NULL 和 IS NOT 
NULL 来 判断 。 注 意 : NULL 和 ' NULL' 是 不 同 的 ， 前 者 表示 为 空 值 ， 后 者 表示 一 个 由 4 个 字母 组 
成 的 字符 串 


运算 符 BETWEEN AND 
BETWEEN AND 用 于 判断 数据 是 否 在 某 个 取 值 范围 内 ， 其 表达 式 如 下 。 
x1 BETWEEN m AND n 
如 果 xl 大 于 等 于 mm， 且 小 于 等 于 n， 结 果 将 返回 1， 否 则 将 返回 0。 
例 7.6 运用 运算 符 BETWEEN AND 判断 数据 表 tb_book 中 的 row 字段 值 是 否 在 10~50 及 25~28 
之 间 ， 查 询 结果 如 图 7.6 所 示 。 实例 位 置 : 光盘 \TMsIV7\7.6) 


t row.row BETWEEN 18 RND 50,row BETWEEN 25 RND 28 from th_hook; 





18 RND 59 


Pows in se 


ysql> 





图 7.6 使 用 运算 符 BETWEEN AND 判断 row 字段 值 的 范围 
从 查询 结果 中 可 以 看 出 ， 在 范围 内 则 返回 1， 否 则 返回 0， 空 值 返回 NULL。 























106 


第 7 章 MySQL 基础 


6. 运算 符 IN 

IN 用 于 判断 数据 是 否 存在 于 某 个 集合 中 ， 其 表达 式 如 下 。 

x1 IN( 值 1, 值 2…' 值 m) 

如 果 xl 等 于 值 1 到 值 n 中 的 任何 一 个 值 ， 结 果 将 返回 1; 如果 不 是 ， 结 果 将 返回 0。 

例 7.7 下 面 运用 运算 符 IN 判断 数据 表 tb_book 中 的 row 字段 值 是 否 在 指定 的 范围 内 ， 查 询 结果 
如 图 7.7 所 示 。 【实例 位 置 : 光盘 \TMNsIN7\7.7) 


lect row,.row INC19-95 -39》 PROM TB_BOOK; 




















row INC19.9 


s in 





图 7.7 使 用 运算 符 IN 判断 row 字段 值 的 范围 

查询 结果 如 图 7.7 所 示 ， 在 范围 内 则 返回 1， 否 则 返回 0， 空 值 返回 NULL。 

7. 运算 符 LIKE 

LIKE 用 来 匹配 字符 串 ， 其 表达 式 如 下 。 

x1 LIKE s1 

如 果 xl 与 字符 串 sl 匹配 ， 结 果 将 返回 1; 否则 返回 0。 

例 7.8 使 用 运算 符 LIKE 判断 数据 表 tb_book 中 的 user 字段 值 是 否 与 指定 的 字符 串 匹 配 ， 查 询 结 
果 如 图 7.8 所 示 。 《实例 位 置 : 光盘 \TMNsI\7\7.8) 


FROM TB_BOOK; 








图 7.8 使 用 运算 符 LIKE 判断 user 字段 是 否 匹 配 某 字符 


查询 结果 如 图 7.8 所 示 ，user 字段 值 为 mr 字符 的 记录 ， 结 果 则 返回 1， 否 则 返 
包含 1 字符 的 记录 ， 匹 配 则 返回 1， 和 否则 返回 0。 








0; user 字段 值 中 
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8. 运算 符 REGEXP 
REGEXP 同样 用 于 匹配 字符 串 ， 但 其 使 用 的 是 正则 表达 式 进行 匹配 ， 其 表达 式 如 下 。 
x1 REGEXP ' 匹 配方 式 ' 


如 果 xl 满足 匹配 方式 ， 结 果 将 返回 1; 否则 将 返回 0。 


























例 7.9 ”使 用 运算 符 REGEXP 来 匹配 user 字段 的 值 是 否 以 指定 字符 开头 、 结 尾 ， 同 时 是 否 包含 指 
定 的 字符 串 ， 执 行 结果 如 图 7.9 所 示 。 《实例 位 置 ， 光盘 \TMNsIN7\7.9) 


-user REGEXP’m’ ,user regexp er regexp’“m’ FROM TB_BOOK; 


nm’ ! user regexp’g$’ ! 


6 rows in set (8.80 sec) 


nysql> , 





图 7.9 使 用 REGEXP 运算 符 匹配 字符 串 
本 例 使 用 运算 符 REGEXP 判断 数据 表 tb_book 表 中 的 user 字段 值 是 否 以 m 字符 开头 ; 是 否 以 g 字 
符 结尾 ， 在 user 字段 值 中 是 否 包含 m 字符 ， 如 果 满足 条 件 则 返回 1， 否 则 返回 0。 
wa 
说 明 
使 用 运算 符 REGEXP 匹配 字符 串 ， 其 使 用 方法 非常 简单 。REGEXP 运算 符 经 常 与 “^” “$? 
和 “.” 一 起 使 用 。“^” 用 来 匹配 字符 串 的 开始 部 分 ; “$” 用 来 匹配 字符 串 的 结尾 部 分 ; “.” 用 
来 代表 字符 串 中 的 一 个 字符 。 


7.1.3 ”逻辑 运算 符 











逻辑 运算 符 用 来 判断 表达 式 的 真 假 。 如 果 表 达 式 是 真 , 结果 返回 1; 如 果 表 达 式 是 假 , 结果 返回 0。 
逻辑 运算 符 又 称 为 布尔 运算 符 。MySQL 中 支持 4 种 逻辑 运算 符 ， 分 别 是 与 、 或 、 非 和 异 或 。 如 表 7.3 
所 示 为 4 种 逻辑 运算 符 的 符号 及 作用 。 











表 7.3 ”逻辑 运算 符 

















符 ”号 作 用 
&& 或 AND 与 
| 或 OR 或 
! 或 NOT 非 
XOR 异 或 
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1. 与 运算 





“&&” 或 者 AND 是 与 运算 的 两 种 表达 方式 。 如 果 所 有 数据 不 为 0 且 不 为 空 值 (NULL) 时 ， 结 

果 返 回 1; 如果 存在 任何 一 个 数据 为 0 时， 结果 返回 0， 如 果 存 在 一 个 数据 为 NULL 且 没 有 数据 为 0 
时 ， 结 果 返 回 NULL。 与 运算 符 支持 多 个 数据 同时 进行 运算 。 

例 7.10 运用 运算 符 “&&” 判 断 row 字段 的 值 是 否 存在 0 或 者 NULL (“row&&1” (row 字段 

1) 和 “row&&0” (row 字段 值 与 0) ) ， 如 果 存 在 则 返回 1， 否 则 返回 0， 空 值 返回 NULL。 执 

了 结果 如 图 7.10 所 示 。 《实例 位 置 : 光盘 \TMsI7\7.10) 












































PowvFow&&l .row&&B from th_hook; 


rov&&l 1 row&&B 


1 
NULL } 
[| 


a 
NULL 上 





图 7.10 ”使 用 运算 符 && 判 断 数据 
2. 或 运算 


“||” 或 者 OR 表示 或 运算 。 所 有 数据 中 存在 任何 一 个 数据 为 非 0 的 数字 时 ， 结 果 返 回 1; 如 果 数 
据 中 不 包含 非 0 的 数字 ， 但 包含 NULL 时 ， 结 果 返 回 NULL; 如 果 操作 数 中 有 0 时 ， 结 果 返 回 0。 或 
运算 符 “||” 也 可 以 同时 操作 多 个 数据 。 

例 a 运用 运算 符 OR 判断 数据 表 tb_book 中 row 字段 是 否 包含 NULL 或 者 非 0 数字 (“row OR 
1” 和 “Irow OR 0”) 。 执 行 结果 如 图 7.11 所 示 。 (实例 位 置 ， 光盘 \TMNsIM77.11) 


Inysql> select row,row OR 1,row OR @ from th_hook; 


1 row } row 





图 7.11 使 用 运算 符 OR 匹配 数据 


显示 ，“row OR 1” 中 包含 NULL 和 1 这 个 非 0 的 数字 ， 所 以 返回 结果 为 1; “row OR 0” 


结果 














中 包含 非 0 的 数字 、NULL 和 0 的 数字 ， 所 以 返回 NULL 和 1。 
3. 非 运算 











“1!” 或 者 NOT 表示 非 运算 。 通 过 非 运算 ， 将 返回 与 操作 数据 相反 的 结果 。 如 果 操 作 数据 是 非 0 
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的 数字 ， 结 果 返 回 0; 如 果 操 作 数 据 是 0， 结 果 返 回 1， 如 果 操 作 数据 是 NULL， 结 果 返 回 NULL。 
例 7.12 运用 运算 符 “! 判断 tb book 表 中 row 字段 的 值 是 否 为 0 或 者 NULL。 执行 结果 如 图 7.12 
所 示 。《〈 实 例 位 置 : 光盘 \TMsI7\7.12) 


五 
5 





el 


























lect row,.trow from th_book; 


in set 0.80 sec) 





图 7.12 ”使 用 运算 符 “!” 判 断 数据 
结果 显示 ，row 字段 中 值 为 NULL 的 记录 ， 返 回 值 为 NULL， 不 为 0 的 记录 ， 返 回 值 为 0。 
4. 异 或 运算 
XOR 表示 异 或 运算 。 只 要 其 中 任何 一 个 操作 数据 为 NULL 时 ， 结 果 返 回 NULL; 如 果 两 个 操作 数 
都 是 非 0 值 ， 或 者 都 是 0， 则 返回 结果 为 0; 如 果 一 个 为 0， 另 一 个 为 非 0 值 ， 返 回 结果 是 1。 
例 7.13 使 用 运算 符 XOR 判断 数据 表 tb_book 中 row 字段 值 是 否 为 NULL(“row XOR 1” 和 “row 
XOR 0”) 。 执 行 结果 如 图 7.13 所 示 。〔 实 例 位 置 ， 光盘 \TMsI\7\7.13) 


lect row,row XOR 1,row XOR 日 from th_hook; 


row XOR 1 


5 rows in set 《8.60 sec) 


nysql1> 





图 7.13 ”使 用 运算 符 XOR 判断 数据 


结果 显示 ，“row XOR 1” 中 row 字段 中 的 值 为 非 0 数字 和 NULL 值 ， 所 以 返回 值 为 0 和 NULL; 
“row XOR 0” 中 包含 0， 所 以 返回 值 为 1; 而 row 字段 值 为 NULL 的 记录 ， 返 回 值 则 为 NULL。 





























7.1.4 位 运算 符 


位 运算 符 是 在 二 进 制 数 上 进行 计算 的 运算 符 。 位 运算 会 先 将 操作 数 变 成 二 进 制 数 再 进行 位 运算 
然后 再 将 计算 结果 从 二 进 制 数 变 回 十 进 制 数 。MySQL 中 支持 6 种 位 运算 符 ， 分 别 是 按 位 与 、 按 位 或 、 
按 位 取 反 、 按 位 异 或 、 按 位 左 移 和 按 位 右 移 。6 种 位 运算 符 的 符号 及 作用 如 表 7.4 所 示 。 
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表 7.4 位 运算 符 
符 ”号 作 用 

按 位 与 。 进 行 该 运算 时 ,数据 库 系统 会 先 将 十 进 制 数 转换 为 二 进 制 数 。 然 后 对 应 操作 数 的 每 个 二 进 制 位 
上 进行 与 运算 。1 和 1 相 与 得 1， 与 0 相 与 得 0。 运 算 完成 后 再 将 二 进 制 数 变 回 十 进 制 数 
按 位 或 。 将 操作 数 化 为 二 进 制 数 后 ， 每 位 都 进行 或 运算 。1 和 任何 数 或 运算 的 结果 都 是 1，0 与 0 或 运 
算 的 结果 为 0 
~ 按 位 取 反 。 将 操作 数 化 为 二 进 制 数 后 ， 每 位 都 进行 取 反 运算 。1 取 反 后 变 成 0，0 取 反 后 变 成 1 
按 位 异 或 。 将 操作 数 化 为 二 进 制 数 后 ， 每 位 都 进行 异 或 运算 。 相 同 的 数 异 或 的 结果 是 0， 不 同 的 数 异 或 
的 结果 为 1 
按 位 左 黎 。“m<<n” 表 示 mm 的 二 进 制 数 向 左 移 n 位 ， 右 边 补 上 mn 个 0。 例 如， 二 进 制 数 001 左 移 1 位 
后 将 变 成 0010 

SS 按 位 右 移 。“m>>n” 表 示 的 二 进 制 数 向 右 移 n 位 ， 左 边 补 上 1n 个 0。 例如， 二 进 制 数 011 右 移 1 位 

后 变 成 001， 最 后 一 个 1 直接 被 移出 

















<< 








例 7.14 将 数字 4 和 6 进行 按 位 与 、 按 位 或 ， 并 将 4 按 位 取 反 。 执 行 结 果 如 图 7.14 所 示 。〔 实 例 
位 置 ， 光盘 \TMNsI\7\7.14) 


set 《0.83 sec)» 





图 7.14 位 运算 的 实例 
7.1.5 ”运算 符 的 优先 级 


由 于 在 实际 应 用 中 可 能 需要 同时 使 用 多 个 运算 符 。 这 就 必须 考虑 运算 符 的 运算 顺序 。 正 所 谓 : 闻 
道 有 先后 ， 术 业 有 专攻 。 

本 节 将 具体 曾 述 MySQL 运算 符 使 用 的 优先 级 ， 如 表 7.5 所 示 。 按 照 从 高 到 低 ， 从 左 到 右 的 级 别 进 
行 运算 操作 。 如 果 优 先 级 相同 ， 则 表达 式 左 边 的 运算 符 先 运算 。 


表 7.5 MySQL 运算 符 的 优先 级 


优 先 级 运 算 符 














111 


MySQL 从 入 门 到 精通 





























优 先 级 运 算 符 
4 */DIV.%.MOD 
和 +,- 
6 >>,.<< 
全 & 
8 | 
学 =,<=>,<,<=,>,>=,!=,<,IN.IS,NULL.LIKE.REGEXP 
10 BETWEEN AND.CASE.WHEN.THEN.ELSE 
11 NOT 
12 &&.AND 
13 I.OR.XOR 
14 = 


7.2 ”流程 控制 语句 


在 MySQL 中 ， 常 见 的 过 程式 SQL 语句 可 以 用 在 一 个 存储 过 程 体 中 。 其 中 包括 正 语句 、CASE 语 
句 、LOOP 语句 、WHILE 语句 、ITERATE 语句 和 LEAVE 语句 ， 它 们 可 以 进行 流程 控制 。 


7.2.1 IF 语句 


正 语句 用 来 进行 条 件 判 断 ， 根 据 不 同 的 条 件 执行 不 同 的 操作 。 该 语句 在 执行 时 首先 判断 正 后 的 条 
件 是 否 为 真 ， 则 执行 THEN 后 的 语句 ， 如 果 为 假 则 继续 判断 正 语句 直到 为 真 为 止 ， 当 以 上 都 不 满足 时 
则 执行 ELSE 语句 后 的 内 容 。 正 语句 表示 形式 如 下 。 


IF condition THEN 
[ELSE 人 THEN] 
ee 

SF 


例 7.15 下 面 通 过 直 …then…else 结构 首先 判断 传 入 参数 的 值 是 否 为 1， 如 果 是 则 输出 1， 如 果 不 
是 则 再 判断 该 传 入 参数 的 值 是 否 为 2， 如 果 是 则 输出 2， 当 以 上 条 件 都 不 满足 时 输出 3。 其 代码 如 下 。 
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(实例 位 置 ， 光盘 \TMNsI\7\7.15) 


delimiter // 
create procedure example_if(in x int) 
begin 

if x=1 then 
select 1; 
elseif x=2 then 
select 2; 

else 

select 3; 
end 所 

end 

I 


以 上 代码 的 运行 结果 如 图 7.15 所 示 。 





procedure example_ifCin x int) 


Ffected (0.00 sec) 





图 7.15 应 用 正 语句 的 存储 过 程 
通过 MySQL 调用 该 存储 过 程 。 其 运行 结果 如 图 7.16 所 示 


1> call example_if (2)// 


in set 0.80 sec> 


IlQuery OK, @ rows affected (‘8.88 sec» 





图 7.16 调用 example_if0) 存 储 过 程 
7.2.2 GASE 语句 


CASE 语句 为 多 分 支 语 句 结构 ， 该 语句 首先 从 WHEN 后 的 VALUE 中 查找 与 CASE 后 的 VALUE 
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相等 的 值 ， 如 果 查 找到 则 执行 该 分 支 的 内 容 ， 否 则 执行 ELSE 后 的 内 容 。CASE 语句 表示 形式 如 下 。 


CASE value 
WHEN value THEN … 
[WHEN valueTHEN…] 
[ELSE.…] 

END CASE 





其 中 ，value 参数 表示 条 件 判断 的 变量 ，WHEN…THEN 中 的 value 参数 表示 变量 的 取 值 。 





CASE 语句 另 一 种 语法 表示 形式 如 下 。 


CASE 
WHEN value THEN… 
[WHEN valueTHEN…] 
[ELSE.…] 

END CASE 


例 7.16 下 面 通过 CASE 语句 首先 判断 传 入 参 类 





下 。《〈 实 例 位 置 : 光盘 \TMNsN7\7.16) 


delimiter // 

create procedure example_casel(in x int) 
begin 

Case X 

when 1 then select 1; 

when 2 then select 2; 

else select 3; 

end case; 

end 

ll 


运行 该 示例 的 结果 如 图 7.17 所 示 。 


Inysql> deliniter // 


create procedure exanple_caselin x int) 


case x 
> when 1 then select 1; 
> when 2 then select 2; 


else lect 33 
end c 
enda 


IlQuery 0K。 日 rows affected 8.88 sec> 


图 7.17 应 用 CASE 语句 的 存储 过 程 


调用 该 存储 过 程 ， 结果 如 图 7.18 所 示 。 
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的 值 是 否 为 1, 如 果 条 件 成 立 则 输出 1, 如 果 条 件 
不 成 立 则 再 判断 该 传 入 参数 的 值 是 否 为 2， 如果 成 立 则 输出 2， 当 以 上 条 件 都 不 ; 


消 足 时 输出 3。 代 码 如 
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call exanple_c 


row in set 《8- 


uery OK。 日 row ected 《8.B1 sec? 





图 7.18 调用 example_case0 存 储 过 程 
7.2.3 WHILE 循环 语句 


WHILE 循环 语句 执行 时 首先 判断 condition 条 件 是 否 为 真 ， 如 果 是 则 执行 循环 体 ， 否 则 退出 循环 。 
去 语句 表示 形式 如 下 。 


WHILE condition DO 
END WHILE; 


例 7.17 下 面 应 用 WHILE 语句 求 前 100 项 的 和 。 首 先 定义 变量 i 和 s， 分 别 用 来 控制 循环 的 次 数 
和 保存 前 100 项 和 ， 当 变量 i 的 值 小 于 或 等 于 100 时 ,使 s 的 值 加 i， 并 同时 使 i 的 值 增 1。 直 到 i 大 于 
100 时 退出 循环 并 输出 结果 。 其 代码 如 下 所 示 。〔 实 例 位 置 ， 光盘 \TMsIN7\7.17) 


delimiter // 

create procedure example_while (out sum int) 
begin 

declare i int default 1; 

declare s int default 0; 

while i<= Pl do 








Set i=i+1; 
end while; 
set sum=s; 
end 

Uh 


运行 以 上 代码 的 结果 如 图 7.19 所 示 。 


sun int》 





图 7.19 应 用 WHILE 语句 的 存储 过 


lls 
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调用 该 存储 过 程 ， 调 用 语句 如 下 所 示 。 


call example_while(@s) 
mysql>select @s 


调用 该 存储 过 程 的 结果 如 图 7.20 所 示 。 


row in set 《8.88 sec> 


nysql> ~ 





图 7.20 调用 example_while0 存 储 过 程 
7.2.4 ”LOOP 循环 语句 


该 循环 没有 内 置 的 循环 条 件 ， 但 可 以 通过 LEAVE 语句 退出 循环 。LOOP 语句 表示 形式 如 下 。 

LOOP 

END LOOP 

LOOP 允许 某 特定 语句 或 语句 群 的 重复 执行 ， 实 现 一 个 简单 的 循环 构造 ， 中 间 省 略 的 部 分 是 需要 
重复 执行 的 语句 。 在 循环 内 的 语句 一 直 重 复 直 至 循环 被 退出 ， 退 出 循环 应 用 LEAVE 语句 。 

LEAVE 语句 经 常 和 BEGIN…END 或 循环 一 起 使 用 ， 其 表示 形式 如 下 。 

LEAVE label 

label 是 语句 中 标注 的 名 字 ， 这 个 名 字 是 自 定义 的 。 加 上 LEAVE 关键 字 就 可 以 用 来 退出 被 标注 的 
循环 语句 。 

例 7.18 ”下面 应 用 LOOP 语句 求 前 100 项 的 和 。 首先 定义 变量 i 和 s, 分 别 用 来 控制 循环 的 次 数 和 
保存 前 100 项 的 和 , 进入 该 循环 体 后 首先 使 s 的 值 加 i, 之 后 使 1 加 1 并 进入 下 次 循环 , 直到 i 大 于 100， 
通过 LEAVE 语句 退出 循环 并 输出 结果 。 其 代码 如 下 。 (实例 位 置 : 光盘 \TMNsN7\7.18) 








delimiter // 

create procedure example_loop (out sum int) 
begin 

declare i int default 1; 

declare s int default 0; 

loop_label:loop 

set s=s+i; 

Set i=i+1; 

ifi>100 then 
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leave loop_label; 
end 所 

end loop; 

set sum=s; 

end 

I 


述 代码 的 运行 结果 如 图 7.21 所 示 。 


ure exanple_loop Cout sum int》 


e i int default 15 
int default @; 


if i>168 then 
> leave loop_label; 
> end 
> end 
> sets 
> end 
> 





lQuery OK, 8 rows affec 


图 7.21 应 用 LOOP 语句 创建 存储 过 程 
调用 名 称 为 example_loop 的 存储 过 程 ， 其 代码 如 下 。 


call example_loop(@s) 
select @s 


运行 结果 如 图 7.22 所 示 。 


-98 sec》 


! 5859 ! 
1 row in set (0.88 sec》> 





图 7.22 调用 example_loop0 存 储 过 程 
7.2.5 REPEAT 循环 语句 


该 语句 先 执行 一 次 循环 体 ， 之 后 判断 condition 条 件 是 否 为 真 ， 为 真 则 退出 循环 ， 否 则 继续 执行 循 
环 。REPEAT 语句 表示 形式 如 下 。 
REPEAT 


UNTIL condition 
END REPEAT 


th 
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例 7.19 下 面 应 用 REPEAT 语句 求 前 100 项 和 。 首 先 定义 变量 i 和 s， 分 别 用 来 控制 循环 的 次 数 
和 保存 前 100 项 和 ,进入 循环 体 后 首先 使 s 的 值 加 i, 之 后 使 1 的 值 加 1， 直 到 i 大 于 100 时 退出 循环 并 
输出 结果 。《〔 实 例 位 置 ， 光盘 \TMNsIN7\7.19) 


delimiter // 

create procedure example_repeat (out sum int) 
begin 

declare i int default 1; 
declare s int default 0; 
repeat 

set s=S+i; 

set i=i+1; 

until i>100 

end repeat; 

set sum=s; 

end 

I 


以 上 代码 的 运行 结果 如 图 7.23 所 示 。 





> deliniter // 
create procedure exanple_repeat Cout sum int? 


begin 
dec i int default 1; 
declam default 0; 


uery OK, @ rows affected 〈 昌 .69 





图 7.23 应 用 REPEAT 语句 创建 存储 过 程 
调用 该 存储 过 程 ， 相 关 代 码 如 下 所 示 。 


call example_repeat(@s) 
select @s 


调用 该 存储 过 程 的 运行 结果 如 图 7.24 所 示 。 





图 7.24 调用 example repeat0 存 储 过 程 
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循环 语句 中 还 有 一 个 ITERATE 语句 ， 它 可 以 出 现在 LOOP、REPEAT 和 WHILE 语句 内 ， 其 意 为 
“再 次 循环 ”。 该 语句 格式 如 下 。 
ITERATE label 
该 语句 的 格式 与 LEAVE 大 同 小 异 ， 区 别 在 于 : LEAVE 语句 是 离开 一 个 循环 ， 而 ITERATE 语句 
是 重新 开始 一 个 循环 。 


注意 
与 一 般 程序 设计 流程 控制 不 同 的 是 : 存储 过 程 并 不 支持 FOR 循环 。 








7:3。 小 结 


本 章 对 MySQL 的 运算 符 和 流程 控制 语句 进行 了 详细 讲解 ， 并 通过 举例 说 明 ， 帮 助 读者 更 好 地 理 
解 所 学 知识 的 用 法 。 在 阅读 本 章 时 ， 读 者 应 该 重点 掌握 各 种 运算 符 和 流程 控制 语句 的 使 有 用， 其中， 位 
运算 符 是 本 章 的 难点 。 因 为 ， 位 运算 符 需 要 将 操作 数 转 换 为 二 进 制 数 ， 然 后 进行 位 运算 。 这 要 求 读者 
能 够 掌握 二 进 制 运 算 的 相关 知识 。 


7.4 实践 与 练习 


1. 编写 SQL 语句 , 将 数字 2、0 和 null 之 间 的 任意 两 个 进行 逻辑 运算 。 (答案 位 置 : 光盘 \IM\ 
SI7\7.20) 
2. 应 用 WHILE 语句 求 前 10 项 的 和 。 (答案 位 置 ， 光盘 \TMNsI7\7.21) 
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表 数 据 的 增 、 删 、 改 操作 


(名 s 视频 讲解 : 20 分 钟 ) 


成 功 创建 数据 库 和 数据 表 以 后 ， 就 可 以 针对 表 中 的 数据 进行 各 种 交互 操作 了 。 
这 些 操作 可 以 有 效 地 使 用 . 维护 和 管理 数据 库 中 的 表 数 据 , 其 中 最 常用 的 就 是 添加 、 
修改 和 删除 操作 了 。 本 章 将 详细 介绍 如 何 通过 SQL 语句 来 实现 表 数 据 的 增 、 删 和 改 
操作 。 

通过 阅读 本 章 ， 读 者 可 以 : 

WI 掌握 插入 表 数 据 的 方法 

MW 掌握 修改 表 中 数据 的 方法 

MW 掌握 删除 表 数 据 的 两 种 方法 


第 8 章 表 数据 的 增 、 删 、 改 操作 


8.1 插入 数据 


在 建立 一 个 空 的 数据 库 和 数据 表 时 ， 首 先 需要 考虑 的 是 如 何 向 数据 表 中 添加 数据 ， 该 操作 可 以 使 
用 INSERT 语句 来 完成 。 使 用 INSERT 语句 可 以 向 一 个 已 有 数据 表 中 插 一 个 新 行 ， 也 就 是 插入 一 行 新 
记录 。 在 MySQL 中 ，INSERT 语句 有 3 种 语法 格式 ， 分 别 是 INSERT…VALUES 语句 、INSERT…SET 
语句 和 INSERT…SELECT 语句 。 下 面 将 分 别 进行 介绍 。 


8.1.1 使 用 INSERT…VALUES 语句 插入 数据 


使 用 INSERT…VALUES 语句 插入 数据 ， 是 INSERT 语句 的 最 常用 的 语法 格式 。 它 的 语法 格式 
如 下 。 
INSERT [LOW_PRIORITY | DELAYED | HIGH_PRIORITY] [IGNORE] 
[INTO] 数据 表 名 [( 字 段 名 ,…)] 
VALUES ({ 值 | DEFAULT},…),(…),… 
[ON DUPLICATE KEY UPDATE 字段 名 = 表达 式 ,…] 


参数 说 明 如 下 。 

(1) [LOW_PRIORITYIDELAYEDIHIGH PRIORITY]: 可 选项 , 其 中 , LOW_PRIORITY 是 INSERT、 
UPDATE 和 DELETE 语句 都 支持 的 一 种 可 选修 饰 符 ， 通常 应 用 在 多 用 户 访问 数据 库 的 情况 下 ,用 于 指 
示 MySQL 降低 INSERT、DELETE 或 UPDATE 操作 执行 的 优先 级 ; DELAYED 是 INSERT 语句 支持 
的 一 种 可 选修 饰 符 ， 用 于 指定 MySQL 服务 器 把 待 插 入 的 行 数 据 放 到 一 个 缓冲 器 中 ， 直 到 待 插 数 据 的 
表 空闲 时 ， 才 真正 在 表 中 插入 数据 行 ，HIGH_PRIORITY 是 INSERT 和 SELECT 语句 支持 的 一 种 可 选 
修饰 符 ， 用 于 指定 INSERT 和 SELECT 操作 优先 执行 的 。 

(2) [IGNORE]: 可 选项 ， 表 示 在 执行 INSERT 语句 时 ， 所 出 现 的 错误 都 会 被 当 作 和 警告 处 理 。 

(3) [INTO] 数据 表 名 : 可 选项 ， 用 于 指定 被 操作 的 数据 表 。 

(4) [( 字 段 名 ,…)]: 可 选项 ， 当 不 指定 该 选项 时 ， 表 示 要 向 表 中 所 有 列 插入 数据 ， 否 则 表示 向 数 
据 表 的 指定 列 插入 数据 。 

(5) VALUES ({ 值 | DEFAULT},…),(…),…: 必 选 项 ， 用 于 指定 需要 插入 的 数据 清单 ， 其 顺序 必须 
与 字段 的 顺序 相对 应 。 其 中 的 每 一 列 的 数据 可 以 是 一 个 常量 、 变 量 、 表 达 式 或 者 NULL， 但 是 其 数据 
类 型 要 与 对 应 的 字段 类 型 相 匹配 ;也 可 以 直接 使 用 DEFAULT 关键 字 ， 表 示 为 该 列 插入 默认 值 ， 但 是 
使 用 的 前 提 是 已 经 明确 指定 了 默认 值 ， 否 则 会 出 错 。 

(6) ON DUPLICATE KEY UPDATE 子 句 : 可 选项 , 用 于 指定 向 表 中 插入 行 时 , 如 果 导 致 UNIQUE 
KEY 或 PRIMARY KEY 出 现 重复 值 ， 系 统 会 根据 UPDATE 后 的 语句 修改 表 中 原 有 行 数据 。 

JINSERT…VALUES 语句 在 使 用 时 ， 通 常 有 以 下 3 种 方式 。 
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1. 插入 完整 数据 


例 8.1 通过 INSERT…VALUES 语句 向 数据 表 tb admin 中 插入 一 条 完整 的 数据 。 (实例 位 置 


盘 \TMNsl\8\8.1) 
(1) 在 编写 SQL 语句 之 前 ， 先 查看 一 下 数据 表 tb_admin 的 表 结 构 ， 具 体 代 码 如 下 。 
DESC db_database08.tb_admin; 
运行 效果 如 图 8.1 所 示 。 





Inysql> DESC db_database8 .th_adnin; 


1 Null 


NO 
NO 





图 8.1 查看 数据 表 tb_admin 的 表 结构 


(2) 编写 SQL 语句 ， 先 选择 数据 表 所 在 的 数据 库 ， 然 后 再 应 用 INSERT…VALUES 语句 实现 向 数 


据 表 tb_admin 中 插入 一 条 完整 的 数据 ， 具 体 代码 如 下 。 


USE db_database08; 
INSERT INTO tb_admin VALUES(1,mr, mrsoft,'2014-09-05 10:25:20"); 


运行 效果 如 图 8.2 所 示 。 


@ 插入 一 条 完整 数据 





图 8.2 向 数据 表 tb_admin 中 插入 一 条 完整 的 数据 
(3) 通过 SELECT * FROM tb_admin 语句 来 查看 数据 表 tb_admin 中 的 数据 ， 有 具体 代码 如 下 。 
SELECT * FROM tb_admin; 
执行 效果 如 图 8.3 所 示 。 











图 8.3 查看 新 插入 的 数据 
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2. 插入 数据 记录 的 一 部 分 


通过 INSERT…VALUES 语句 还 可 以 实现 向 数据 表 中 插入 数据 记录 的 一 部 分 , 也 就 是 只 插入 表 的 一 
行 中 的 某 几 个 字段 的 值 ， 下 面 通过 一 个 具体 的 实例 来 演示 如 何 向 数据 表 中 插入 数据 记录 的 一 部 分 。 还 








是 以 例 8.1 中 使 用 的 数据 表 tb_admin 为 例 进行 插入 。 





例 8.2 通过 INSERT…VALUES 语句 向 数据 表 tb_admin 中 插入 数据 记录 的 一 部 分 。 (实例 位 置 ; 


光盘 \TMsIN8\8.2) 





(1) 编写 SQL 语句 ， 先 选择 数据 表 所 在 的 数据 库 ， 然 后 再 应 用 INSERT…VALUES 语句 实现 向 数 


据 表 tb_admin 中 插入 一 条 记录 ， 只 包括 user 和 password 字段 的 值 ， 具 体 代码 如 下 。 





USE db_database08; 
INSERT INTO tb_admin (user,password) 
VALUES(rjkfltm', 1117; 


运行 效果 如 图 8.4 所 示 。 


SC' zjkfln' 111 


row affected 〈 昌 .898 sec)» 





图 8.4 向 数据 表 tb_admin 中 插入 数据 记录 的 一 部 分 
(2) 通过 SELECT * FROM tb_admin 语句 来 查看 数据 表 tb_admin 中 的 数据 ， 有 具体 代码 如 下 。 
SELECT * FROM tb_admin; 
执行 效果 如 图 8.5 所 示 。 


mysql> SELECT # FROM thb_adnin; 


user 了 ord ! createtime 


mm soft 14-89-85 1 


jkf ln 1 111 


set 0.80|sec> 


id 为 2 的 行 是 新 插入 的 数据 








8.5 查看 新 插入 的 数据 











/ 
说 明 


由 于 在 设计 数据 表 时 ， 将 id 字段 设置 为 自动 编号 ， 所 以 即使 没有 指定 id 的 值 ，MySQL 也 会 


自动 为 它 填 上 相应 的 编号 。 


MySQL 从 入 门 到 精通 


3. 插入 多 条 记录 


通过 INSERT…VALUES 语句 还 可 以 实现 一 次 性 插入 多 条 数据 记录 。 使 用 该 方法 批量 插入 数据 ， 比 
使 用 多 条 单行 的 INSERT 语句 的 效率 要 高 。 下 面 将 通过 一 个 具体 的 实例 演示 如 何 一 次 插入 多 条 记录 。 
例 8.3 ”通过 INSERT…VALUES 语句 向 数据 表 tb_admin 中 一 次 插入 多 条 记录 。( 实 例 位 置 : 光盘 
\TMN\sN\8\8.3) 
(1) 编写 SQL 语句 ， 先 选择 数据 表 所 在 的 数据 库 ， 然 后 再 应 用 INSERT…VALUES 语句 实现 向 数 
据 表 tb_admin 中 插入 3 条 记录 ， 都 只 包括 user、password 和 createtime 字段 的 值 ， 具 体 代 码 如 下 。 
USE db_database08; 
INSERT INTO tb_admin (user,password,createtime) 
VALUES('mrbcced','111', '2014-09-05 10:35:26') 
,( 'mingri',"111", '2014-09-05 10:45:27") 
,( 'mingrisoft’'111", '2014-09-05 10:55:28"); 








运行 效果 如 图 8.6 所 示 。 


RT INIO thb_adn 
‘mrbhcc :111 
hk ke 14-89-05 10:4 


oft” 14-89-85 19 
aff 《B .DB 
Duplicat Warn in 


-次 性 插入 3 条 记录 





图 8.6 向 数据 表 tb_admin 中 插入 3 条 记录 
(2) 通过 SELECT * FROM tb_admin 语句 来 查看 数据 表 tb_admin 中 的 数据 ， 有 具体 代码 如 下 。 
SELECT * FROM tb_admin; 
执行 效果 如 图 8.7 所 示 。 


Inysql> SELECT * FROM th_adnin; 


1 user 


89-85 


set 《0.60 s 














8.7 查看 新 插入 的 3 行 数据 
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8.1.2 ”使 用 INSERT…SET 语句 插入 数据 


在 MySQL 中 ， 除 了 可 以 使 用 INSERT-…VALUES 语句 插入 数据 外 ， 还 可 以 使 用 INSERT…SET 语 
句 插 入 数据 。 这 种 语法 格式 用 于 通过 直接 给 表 中 的 某 些 字段 指定 对 应 的 值 来 实现 插入 指定 数据 ， 对 于 
未 指定 值 的 字段 将 采用 默认 值 进行 添加 。INSERT…SET 语句 的 语法 格式 如 下 。 








INSERT [LOW_PRIORITY | DELAYED | HIGH_PRIORITY] [IGNORE] 
[INTO] 数据 表 名 
SET 字段 名 ={ 值 | DEFAULT}, … 
[ ON DUPLICATE KEY UPDATE 字段 名 = 表达 式 ,…] 


参数 说 明 如 下 。 

(1) [LOW_PRIORITY | DELAYED | HIGH PRIORITY] [IGNORE]: 可 选项 ， 其 作用 与 
INSERT…VALUES 语句 相同 ， 这 里 将 不 再 歼 述 。 

(2) [INTO] 数据 表 名 : 用 于 指定 被 操作 的 数据 表 ， 其 中 ，[INTO] 为 可 选项 ， 可 以 省 略 。 

(3) SET 字段 名 ={ 值 |DEFAULT}: 用 于 给 数据 表 中 的 某 些 字段 设置 要 插入 的 值 。 

(4) ON DUPLICATE KEY UPDATE 子 句 : 可 选项 ， 其 作用 与 INSERT…VALUES 语句 相同 ， 这 
里 将 不 再 资 述 。 

例 8.4 通过 INSERT…SET 语句 向 数据 表 tb_admin 中 插入 一 条 记录 。( 实 例 位 置 :光盘 \TMNsl\8\8.4) 

(1) 编写 SQL 语句 ， 先 选择 数据 表 所 在 的 数据 库 ， 然 后 再 应 用 INSERT…SET 语句 实现 向 数据 表 
tb_admin 中 插入 一 条 记录 ， 包 括 user、password 和 createtime 字段 的 值 ， 具 体 代 码 如 下 。 

USE db_database08; 


INSERT INTO tb_admin 
SET user='mrbccd',password='111',createtime= '2014-09-06 10:35:26'; 


运行 效果 如 图 8.8 所 示 。 


rp @ 选择 数据 表 所 在 的 数据 库 


T INIO tb_adm 
= mrbccd'” , word=’ ?oo@ ime= ’2814-89-86 19:3 
Query OK, 1 row affected (8.064 


nysql> ,~ @ 插 入 一 条 记录 





图 8.8 向 数据 表 tb_admin 中 插入 一 条 记录 
(2) 通过 SELECT * FROM tb_admin 语句 来 查看 数据 表 tb_admin 中 的 数据 ， 有 具体 代码 如 下 。 
SELECT * FROM tb_admin; 
执行 效果 如 图 8.9 所 示 。 
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CT * FROM th_adnin; 


新 插入 的 一 行 数据 





图 89 查看 新 插入 的 一 行 数据 
8.1.3 插入 查询 结果 


在 MySQL 中 , 支持 将 查询 结果 插入 到 指定 的 数据 表 中 ,这 可 以 通过 INSERT…SELECT 语句 来 
实现 。 
INSERT [LOW_PRIORITY | HIGH_PRIORITY] [IGNORE] 
[INTO] 数据 表 名 [( 字 段 名 ,…)] 
SELECT … 
[ON DUPLICATE KEY UPDATE 字段 名 = 表达 式 , … ] 


参数 说 明 如 下 。 

(1) [LOW_PRIORITYIDELAYEDIHIGH_PRIORITY] [IGNORE]: 可 选项 , 其 作用 与 INSERT…VALUES 
语句 相同 ， 这 里 不 再 效 述 。 

(2) [INTO] 数据 表 名 : 用 于 指定 被 操作 的 数据 表 ， 其 中 ，[INTO] 为 可 选项 ， 可 以 省 略 。 

(3) [( 字 段 名 ,…)]: 可 选项 ， 当 不 指定 该 选项 时 ， 表 示 要 向 表 中 所 有 列 插入 数据 ， 否 则 表示 向 数 
昌 表 的 指定 列 插入 数据 。 

(4) SELECT 子 句 : 用 于 快速 地 从 一 个 或 者 多 个 表 中 取出 数据 ， 并 将 这 
目标 数据 表 中 。 需 要 注意 的 是 ，SELECT 子 句 返回 的 结果 集中 的 字段 数 、 字 
完全 一 致 。 

(5) ON DUPLICATE KEY UPDATE 子 句 : 可 选项 ， 其 作用 与 NSERT…VALUES 语句 相同 ， 这 
里 不 再 歼 述 。 

例 8.5 从 数据 表 tb_mrbook 中 查询 出 user 和 pass 字段 的 值 ， 插 入 到 数据 表 tb_admin 中 。 (实例 
位 置 ， 光盘 \TMsl\8\8.5) 

(1) 查看 数据 表 tb_mrbook 的 表 结构 ， 具 体 代码 如 下 。 


DESC db_database08.tb_mrbook:; 








数据 作为 行 数据 插入 到 
段 类 型 必须 与 目标 数据 表 
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执行 效果 如 图 8.10 所 示 。 


1> DESC dhb_data eB8.th_nrhook; 


Default ! 


varchar48> 





图 8.10 数据 表 tb_mrbook 的 表 结 构 
(2) 查询 数据 表 tb_mrbook 中 的 数据 ， 具 体 代 码 如 下 。 
SELECT * FROM tb_mrbook; 
执行 效果 如 图 8.11 所 示 。 
Inysql> SELECT *» FROM db_databa 8 .th_nrbook; 


booknane ! type 


z JAUA 上 PHP 
LI PHP 
ZHAO - 


这 部 分 数据 是 要 插入 到 目标 表 中 的 数据 








图 8.11 数据 表 tb_mrbook 的 数据 


(3) 编写 SQL 语句 ， 实 现 从 数据 表 tb_mrbook 中 查询 user 和 pass 字段 的 值 ， 插 入 到 数据 表 tb_admin 
中 ， 有 具体 代码 如 下 。 


INSERT INTO db_database08.tb_admin 
(user,password) 
SELECT user,pass FROM tb_mrbook; 


执行 效果 如 图 8.12 所 示 。 


T INIO db_database@8 .th_adnin 
assuord) 
ss FROM th_mrbook; 
ed ‘8.84 》 


: 8 Warning 





图 8.12 将 查询 结果 插入 到 数据 表 
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(4) 通过 SELECT 语句 来 查看 数据 表 tb_admin 中 的 数据 ， 具 体 代 码 如 下 。 
SELECT * FROM db_database08.tb_admin; 
执行 效果 如 图 8.13 所 示 。 


SELECT * FROM db_database@8 .th_adnin; 
! password ! crea 
-* 一 一 -一 一 ^ 
mr ! mrsoft 
rjkf lm 1 111 
mrbccd 1 111 
i ! 111 
上 111 
! 111 


19 rows in set 《0.688 secy) 


新 插入 的 4 行 数据 





nysql> 


图 8.13 查看 新 插入 的 数据 
8.2 修改 数据 


要 执行 修改 的 操作 可 以 使 用 UPDATE 语句 ， 语 法 如 下 。 


UPDATE [LOW_PRIORITY] [IGNORE] 数据 表 名 
SET 字段 {= 值 1[, 字段 2= 值 2…] 
[WHERE 条 件 表达 式 ] 

[ORDER BY-…] 
[LIMIT 行 数 ] 


参数 说 明 如 下 。 

(1) [LOW_PRIORITY]: 可 选项 , 表示 在 多 用 户 访问 数据 库 的 情况 下 可 用 于 延 妈 UPDATE 操作 ， 
直到 没有 别 的 用 户 再 从 表 中 读 取 数据 为 止 。 这 个 过 程 仅 适用 于 表 级 锁 的 存储 引擎 (如 IyISAM、 
MEMORY 和 MERGE) 。 

(2) [IGNORE]: 在 MySQL 中 ， 通 过 UPDATE 语句 更 新 表 中 多 行 数据 时 ， 如 果 出 现 错误 ， 那 么 
整个 UPDATE 语句 操作 都 会 被 取消 ， 错 误 发 生前 更 新 的 所 有 行将 被 恢复 到 它们 原来 的 值 。 因 此 ， 为 了 
在 发 生 错误 时 也 要 继续 进行 更 新 ， 则 可 以 在 UPDATE 语句 中 使 用 IGNORE 关键 字 。 

(3) SET 子 句 : 必 选 项 ， 用 于 指定 表 中 要 修改 的 字段 名 及 其 字段 值 。 其 中 的 值 可 以 是 表达 式 ， 也 
可 以 是 该 字段 所 对 应 的 默认 值 。 如 果 指 定 默认 值 ， 那 么 使 用 关键 字 DEFAULT 指定 。 
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(4) WHERE 子 句 : 可 选项 ， 用 于 限定 表 中 要 修改 的 行 ， 如 果 不 指定 该 子 句 ， 那 么 UPDATE 语 
句 会 更 新 表 中 的 所 有 行 。 
(5) ORDER BY 子 句 : 可 选项 ， 用 于 限定 表 中 的 行 被 修改 的 次 序 。 
(6) LIMIT 子 句 : 可 选项 ， 用 于 限定 被 修改 的 行 数 。 
例 8.6 将 管理 员 信息 表 tb_admin 中 用 户 名 为 mrbccd 的 管理 员 密 码 111 修改 为 123。 (实例 位 置 ; 
光盘 \TMsI8\8.6) 
编写 SQL 语句 修改 用 户 名 为 mrbccd 的 管理 员 密码 为 123， 具 体 代码 如 下 。 
UPDATE db_database08.tb_admin SET password='123' WHERE user='mrbccd '; 


运行 效果 如 图 8.14 所 示 。 





Inysql> UPDATE db_databaseD8 .tb_adnin SET password=’123’ WHERE user='nrbccd'; 
Query 0K。2 rows affected 8.82 sec) 


Rows matched: 2 Changed: 2 Warnings: 


将 mrbccd 用 户 的 密码 更 改 为 123 


nysql> 





图 8.14 将 mrbccd 用 户 的 密码 更 改 为 123 


入 s 注 意 
更 新 时 一 定 要 保证 WHERE 子 句 的 正确 性 ， 一 旦 WHERE 子 名 出 错 ， 将 会 破坏 所 有 改变 的 
数据 。 


查询 修改 后 的 数据 库 内容 ， 代 码 如 下 。 
SELECT * FROM db_database08.tb_admin WHERE user='mrbccd '; 
执行 结果 如 图 8.15 所 示 。 


和 ysal》 SELECT * FROM db_a sel admin WHERE user=’mrbccd’; 


rovs in set 《8.9B sec) 


Inysql> 





图 8.15 查看 修改 后 的 结果 
8.3 删除 数据 


在 数据 库 中 ， 有 些 数据 已 经 失去 意义 或 者 错误 时 就 需要 将 它们 删除 ， 在 MySQL 中 ， 可 以 使 用 
DELETE 语句 或 者 TRUNCATE TABLE 语句 删除 表 中 的 一 行 或 多 行 数据 ， 下 面 分 别 进行 介绍 。 
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8.3.1 通过 DELETE 语句 删除 数据 


通过 DELETE 语句 删除 数据 的 基本 语法 格式 如 下 。 
DELETE [LOW_PRIORITY] [QUICK] [IGNORE] FROM 数据 表 名 
[WHERE 条 件 表达 式 ] 
[ORDER BY…] 
[LIMIT 行 数 ] 

参数 说 明 如 下 。 

(1) [LOW _PRIORITY]: 可 选项 ， 表 示 在 多 用 户 访问 数据 库 的 情况 下 可 用 于 延迟 DELETE 操作 ， 
直到 没有 别 的 用 户 再 从 表 中 读 取 数据 为 止 。 这 个 过 程 仅 适用 于 表 级 锁 的 存储 引擎 (如 IyISAM、 
MEMORY 和 MERGE) 。 

(2) [QUICK]: 可 选项 ， 用 于 加 快 部 分 种 类 的 删除 操作 的 速度 。 

(3) [IGNORE]: 在 MySQL 中 ， 通 过 DELETE 语句 删除 表 中 多 行 数据 时 ， 如 果 出 现 错误 ， 那 么 
整个 DELETE 语句 操作 都 会 被 取消 ， 错 误 发 生前 更 新 的 所 有 行将 被 恢复 到 它们 原来 的 值 。 因 此 ， 为 了 
在 发 生 错误 时 也 要 继续 进行 删除 ， 则 可 以 在 DELETE 语句 中 使 用 IGNORE 关键 字 。 

(4) 数据 表 名 : 用 于 指定 要 删除 的 数据 表 的 表 名 。 

(5) WHERE 子 句 : 可 选项 ,用 于 限定 表 中 要 删除 的 行 ， 如 果 不 指定 该 子 句 ， 那 么 DELETE 语句 
会 删除 表 中 的 所 有 行 。 

(6) ORDER BY 子 句 : 可 选项 ， 用 于 限定 表 中 的 行 被 删除 的 次 序 。 

(7) LIMIT 子 句 : 可 选项 ， 用 于 限定 被 删除 的 行 数 。 


息 o 注 意 


该 语句 在 执行 过 程 中 ， 如 果 没 有 指定 WHERE 条 件 ， 将 删除 所 有 的 记录 ; 如 果 指 定 了 WHERE 
条 件 ， 将 按照 指定 的 条 件 进行 删除 。 


例 8.7 删除 管理 员 数 据 表 tb_admin 中 用 户 名 为 mr 的 记录 信息 。 实例 位 置 ， 光盘 \TMsI\8\8.7) 
(1) 编写 SQL 语句 除 管理 员 数 据 表 tb_admin 中 用 户 名 为 mr 的 记录 信息 ， 有 具体 代码 如 下 。 


USE db_database08; 
DELETE FROM tb_admin WHERE user='mr'; 


运行 效果 如 图 8.16 所 示 。 














图 8.16 ”删除 数据 表 中 用 户 名 为 mr 的 记录 
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(2) 通过 SELECT 语句 来 查看 删除 记录 后 数据 表 tb_admin 中 的 数据 ， 具 体 代码 如 下 。 
SELECT * FROM tb_admin; 
执行 结果 如 图 8.17 所 示 。 








图 8.17 查看 删除 后 的 结果 


删除 数据 前 ， 数 据 表 tb _admin 中 的 数据 如 图 8.13 所 示 。 


《和 注意 
在 实际 的 应 用 中 ,执行 删除 操作 时 ,执行 删除 的 条 件 一 般 应 该 为 数据 的 这， 而 不 是 具体 某 个 字 
段 值 ， 这 样 可 以 避免 一 些 错误 发 生 。 


8.3.2 通过 TRUNCATE TABLE 语句 删除 数据 


在 删除 数据 时 ， 如 果 要 从 表 中 删除 所 有 的 行 ， 通 过 TRUNCATE TABLE 语句 删除 数据 的 基本 语法 
格式 如 下 。 
TRUNCATE [TABLE] 数据 表 名 
在 上 面 的 语法 中 ， 数 据 表 名 表示 的 就 是 删除 的 数据 表 的 表 名 ， 也 可 以 使 用 “数据 库 名 .数据 表 名 ” 
来 指定 该 数据 表 隶 属于 哪个 数据 库 。 
和 注意 
由 于 TRUNCATE TABLE 语句 会 删除 数据 表 中 的 所 有 数据 ， 并 且 无 法 恢复 ， 因 此 使 用 
TRUNCATE TABLE 语句 时 一 定 要 十 分 小 心 。 









例 8.8 使 用 TRUNCATE TABLE 语句 清空 管理 员 数 据 表 tb_admin， 具 体 代 码 如 下 。 (实例 位 置 : 
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光盘 \TMsIN8\8.8) 
TRUNCATE TABLE db_database08.tb_admin; 


运行 效果 如 图 8.18 所 示 。 


Inysql> TRUNCATE TABLE db_databaseB8 .th_adnin; 
jQuery OK, @ rows affected (@.83 sec) 





图 8.18 清空 管理 员 数 据 表 tb_admin 


DELETE 语句 和 TRUNCATE TABLE 语句 的 区 别 如 下 。 

(1) 使 用 TRUNCATE TABLE 语句 后 , 表 中 的 AUTO _INCREMENT 计数 器 将 被 重新 设置 为 该 列 
的 初始 值 。 

(2) 对 于 参与 了 索引 和 视图 的 表 ， 不 能 使 用 TRUNCATE TABLE 语句 来 删除 数据 ， 而 应 用 使 用 
DELETE 语句 。 

(3) TRUNCATE TABLE 操作 与 DELETE 操作 使 用 的 系统 和 事务 日 志 资 源 少 。DELETE 语句 每 
删除 一 行 都 会 在 事务 日 志 中 添加 一 行 记录 ， 而 TRUNCATE TABLE 语句 是 通过 释放 存储 表 数 据 所 用 的 
数据 页 来 删除 数据 的 ， 因 此 只 在 事务 日 志 中 记录 页 的 释放 。 





8.4 小 结 





本 章 介绍 了 在 MySQL 中 向 数据 表 中 添加 数据 库 、 修 改 数据 和 删除 数据 的 具体 方法 ， 也 就 是 对 表 
数据 的 增 、 删 和 改 操 作 。 这 3 种 操作 在 实际 开发 中 经 常 应 用 。 因 此 ， 对 于 本 章 的 内 容 需 要 认真 学 习 ， 
争取 做 到 举一反三 、 灵 活 应 用 。 


8.5 实践 与 练习 


1. 编写 SQL 语句 ， 先 选择 数据 表 所 在 的 数据 库 ， 然 后 再 应 用 INSERT…VALUES 语句 实现 向 数据 
表 tb_admin 中 插入 一 条 记录 ， 只 包括 user、password 和 createtime 字段 的 值 。 (答案 位 置 ， 光盘 
\TMN\sN\8\8.9) 

2. 编写 SQL 语句 ， 使 用 DELETE 语句 清空 管理 员 数据 表 tb_admin。 (答案 位 置 光盘 \TMNsI\8\8.10) 
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第 章 


ee 询 


( 馈 " 视频 讲 分 钟 ) 


数据 查询 是 指 从 数据 库 中 获取 所 需要 的 数据 。 数 据 查 询 是 数据 库 操作 中 最 常 
用 ， 也 是 最 重要 的 操作 。 通 过 不 同 的 查询 方式 可 以 获得 不 同 的 数据 。 用 户 可 以 根据 
自己 对 数据 的 需求 使 用 不 同 的 查询 方式 。 在 My5QL 中 是 使 用 SELECT 语句 来 查询 
数据 的 。 本 章 将 对 查询 语句 的 基本 语法 、 在 单 表 上 查询 数据 、 使 用 聚合 函数 查询 数 
据 、 合 并 查询 结果 等 内 容 进 行 详细 的 讲解 ， 帮 助 读者 轻松 了 解 查 询 数 据 的 语句 。 

通过 阅读 本 章 ， 读 者 可 以 : 

了 解 基本 查询 语句 

了 解 单 表 查 询 的 方法 

了 解 用 聚合 困 数 查询 的 方法 
掌握 连 接 查 询 和 子 查 询 的 方法 
掌握 合并 查询 结果 的 方法 
掌握 定义 表 和 字段 别名 的 方法 
掌握 正则 表达 式 查询 的 方法 


于 于 于 于 至 至 至 


MySQL 从 入 门 到 精通 


9.1 基本 查询 语句 


SELECT 语句 是 最 常用 的 查询 语句 ， 它 的 使 用 方式 有 些 复杂 ， 但 功能 是 相当 强大 的 。SELECT 语 


句 的 基本 语法 如 下 。 
select selection_list /要 查询 的 内 容 ， 选 择 哪些 列 
from 数据 表 名 /指定 数据 表 
where primary_constraint /查询 时 需要 满足 的 条 件 ， 行 必须 满足 的 条 件 
group by grouping_columns /如何 对 结果 进行 分 组 
order by sorting_cloumns // 如 何 对 结果 进行 排序 
having secondary_constraint // 查 询 时 满足 的 第 二 条 件 
limit count // 限 定 输出 的 查询 结果 





其 中 使 用 的 子 句 将 在 后 面 逐 个 介绍 。 下 面 先 介绍 SELECT 语句 的 简单 应 用 。 
1. 使 用 SELECT 语句 查询 一 个 数据 表 
使 用 SELECT 语句 时 ， 首 先 要 确定 所 要 查询 的 列 。“*” 代 表 所 有 的 列 。 例如， 查询 db_database09 


数据 库 user 表 中 的 所 有 数据 ， 代 码 如 下 。 


如 ， 


mysql> use db_database09 

Database changed 

mysql> select * from user; 

十 ~--- 十 ~-- 一 -十 - 十 一 ---- 一 -一 -一 十 

lid | user | Ixdh |jtdz 1 
十 ~--- 十 -~----- 十 - 二 -十 

| 1|mr |12345678 | 长 春 市 | 

| 2|mrsoft |87654321 | 四 平市 | 
十 ---- 十 ------ 十 - 十- 
2 rows in set (0.00 sec) 


这 是 查询 整个 表 中 所 有 列 的 操作 ， 还 可 以 针对 表 中 的 某 一 列 或 多 列 进行 查询 。 
2. 查询 表 中 的 一 列 或 多 列 


针对 表 中 的 多 列 进行 查询 ， 只 要 在 select 后 面 指定 要 查询 的 列 名 即 可 ， 多 列 之 间 用 “,” 分 隔 。 例 
查询 user 表 中 的 id 和 lxdh， 代 码 如 下 。 


mysql> select id, lxdh from user ; 
十 -一 十 - -十 

lid | Ixdh | 

十 一 一 十 - + 

| 11123456781 

| 21876543211 

士 一 一 十 - 二 

2 rows in set (0.00 sec) 


























3. 从 一 个 或 多 个 表 中 获取 数据 


使 用 SELECT 语句 进行 查询 ， 需 要 确定 所 要 查询 的 数据 在 哪个 表 或 哪些 表 中 ， 在 对 多 个 表 进 行 查 
询 时 ， 同 样 使 用 “,” 对 多 个 表 进 行 分 隔 。 

例 9.1 从 tb admin 表 和 tb_students 表 中 查询 出 tb_admin.id、tb_admin.tb user、tb_students.id 和 
tb_students.name 字段 的 值 ， 其 代码 如 下 。 (实例 位 置 : 光盘 \TMNsI\9\9.1) 


mysql> select tb_admin.id,tb_admin.tb_usertb_students.id,tb_students.name from 
tb_admin,tb_students; 

十 -一 十 - 十 -一 十 -一 -一 -十 

lidltb_user |id|name| 

十 -十 -一 -一 十 -十 一- 十 

| 11mr | 11 潘 攀 | 

| 21 明日 科技 | 11 潘 攀 | 

+--—-+- 十 ~--- 十 一- 一 -十 

2 rows in set (0.03 sec) 


mg 
在 查询 数据 库 中 的 数据 时 ， 如 果 数 据 中 涉及 中 文字 符 囊 ,， 有 可 能 在 输出 时 会 出 现 乱 码 。 那么 最 
后 在 执行 查询 操作 之 前 ， 通 过 set names 语句 设置 其 编码 格式 ， 然 后 再 输出 中 文字 符 串 时 就 不 会 出 
现 乱码 了 。 如 本 例 中 所 示 ， 应 用 set names 语句 设置 其 编码 格式 为 gb2312。 








还 可 以 在 WHERE 子 句 中 使 用 连接 运算 来 确定 表 之 间 的 联系 ， 然 后 根据 这 个 条 件 返 回 查询 结果 。 
例如 ， 从 家 庭 收入 表 (jtsr) 中 查询 出 指定 用 户 的 家 庭 收入 数据 ， 条 件 是 用 户 的 功 为 1， 代码 如 下 。 
mysql> select jtsr from userjtsr 


-> Where user.user=jtsr.user and user.id=1 ; 
十 ------ 十 
| jtsr | 
十 ------ 十 


1100001 


十 ~------ 十 
2 rows in set (0.00 sec) 


其 中 , user.user = jtsr.user 将 表 user 和 jtsr 连接 起 来 , 叫 作 等 同 连接 ; 如 果 不 使 用 user.user= jtsr.user， 
那么 产生 的 结果 将 是 两 个 表 的 笛 卡 尔 积 ， 叫 作 全 连接 。 





9.2 单 表 查询 


单 表 查询 是 指 从 一 张 表 中 查询 所 需要 的 数据 ， 所 有 查询 操作 都 比较 简单 。 
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9.2.1 查询 所 有 字段 





查询 所 有 字段 是 指 查询 表 中 所 有 字段 的 数据 。 这 种 方式 可 以 将 表 中 所 有 字段 的 数据 都 查询 出 来 。 
在 MySQL 中 可 以 使 用 “* ”代表 所 有 的 列 ， 即 可 查 出 所 有 的 字段 ， 语 法 格式 如 下 。 

SELECT * FROM 表 名 ; 

其 应 用 已 经 在 9.1 节 基 本 查询 语句 中 介绍 过 ， 这 里 不 再 袭 述 。 








9.2.2 查询 指定 字段 


查询 指定 字段 可 以 使 用 下 面 的 语法 格式 。 

SELECT 字段 名 FROM 表 名 ; 

如 果 是 查询 多 个 字段 ， 可 以 使 用 “,” 对 字段 进行 分 隔 。 

例 9.2 查询 db database09 数据 库 tb login 表 中 user 和 pwd 两 个 字段 ，SELECT 查询 语句 如 下 。 
《实例 位 置 ， 光盘 \TMNsI\9\9.2) 

SELECT user,pwd FROM tb_login; 


查询 结果 如 图 9.1 所 示 。 


n set (0.00 sec》 





图 9.1 查询 指定 字段 的 数据 
9.2.3 ”查询 指定 数据 


如 果 要 从 很 多 记录 中 查询 出 指定 的 记录 ， 那 么 就 需要 一 个 查询 的 条 件 。 设 定 查 询 条 件 应 用 的 是 
WHERE 子 句 。 通 过 它 可 以 实现 很 多 复杂 的 条 件 查询 。 在 使 用 WHERE 子 句 时 ， 需 要 使 用 一 些 比较 运 





算 符 来 确定 查询 的 条 件 。 其 常用 的 比较 运算 符 如 表 9.1 所 示 。 
表 9.1 比较 运算 符 






称 运 算 符 示 例 
= 等 于 Is not null Id is not null 


[ed 
所 
冲 
改 


















Between JIdbetween 1 and 15 
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续 表 
运 算 符 名 称 示 例 运 算 符 名 称 示 例 
< 小 于 id<5 IN ma idin (3.4.5) 
=> 大 于 等 于 id=>5 NOTIN ma name not in (shi.li) 
< 小 于 等 于 id<=5 LIKE 模式 匹配 name like ('shi%%') 
!= 或 二 不 等 于 id!=5 NOT LIKE 模式 匹配 name not like ('shi%%') 
IS NULL ma id is null REGEXP 常规 表达 式 name 正则 表达 式 
表 9.1 中 列举 的 是 WHERE 子 句 常用 的 比较 运算 符 ， 例 中 的 id 是 记录 的 编号 ，name 是 表 中 的 用 
户 名 。 
例 9.3 应 用 WHERE 子 句 查询 tb_login 表 ， 条 件 是 user (用 户 名 ) 为 mr， 代 码 如 下 。【〔 实 例 位 


置 : 


光盘 \TMNsl\9\9.3) 
select * from tb_login where user = 'mr'; 
查询 结果 如 图 9.2 所 示 。 
Inysql> select * from th_login where user=’ 
Soft 1 PHP 


1 row in set 0.80 sec) 


nysal 





图 9.2 查询 指定 数据 


9.2.4” 带 关键 字 IN 的 查询 


关键 字 IN 可 以 判断 某 个 字段 的 值 是 否 在 指定 的 集合 中 。 如 果 字 段 的 值 在 集合 中 , 则 满足 查询 条 件 ， 


该 记录 将 被 查询 出 来 ， 如 果 不 在 集合 中 ， 则 不 满足 查询 条 件 。 其 语法 格式 如 下 。 


SELECT * FROM 表 名 WHERE 条 件 [NOT] IN( 元 素 1, 元 素 2,…, 元 素 n); 

(1) [NOT]: 是 可 选项 ， 加 上 NOT 表示 不 在 集合 内 满足 条 件 ; 

(2) 元 素 : 表示 集合 中 的 元 素 ， 各 元 素 之 间 用 逗号 隔 开 ， 字 符 型 元 素 需要 加 上 单 引号 。 

例 9.4 应 用 IN 关键 字 查询 tb_login 表 中 user 字段 为 mr 和 的 记录 ， 查 询 语句 如 下 。 (实例 位 








: 光盘 \TMNsl\9\9.4) 


SELECT * FROM tb_login WHERE user INCmr,Ix'):; 
查询 结果 如 图 9.3 所 示 。 


137 


MySQL 从 入 门 到 精通 


in set 《08.88 sec) 





图 9.3 使 用 关键 字 IN 查询 





例 9.5 使 用 关键 字 NOT IN 查询 tb login 表 中 user 字段 不 为 mr 和 Ix 的 记录 , 查询 语句 如 下 。( 实 
例 位 置 : 光盘 \TMNsI\9\9.5) 


SELECT* FROM tb_login WHERE user NOT IN('mr','Ix'); 
查询 结果 如 图 9.4 所 示 。 


《0.800 sec》 





图 9.4 使 用 关键 字 NOT IN 查询 


9.2.5” 带 关键 字 BETWEEN AND 的 范围 查询 





关键 字 BETWEEN AND 可 以 判断 某 个 字段 的 值 是 否 在 指定 的 范围 内 。 如 果 字 段 的 值 在 指定 范围 
内 ， 则 满足 查询 条 件 ， 该 记录 将 被 查询 出 来 。 如 果 不 在 指定 范围 内 ， 则 不 满足 查询 条 件 。 其 语法 如 下 。 

SELECT * FROM 表 名 WHERE 条 件 [NOT] BETWEEN 取 值 1 AND 取 值 2; 

(1) [NOT]: 可 选项 ， 表 示 不 在 指定 范围 内 满足 条 件 。 

(2) 取 值 1: 表示 范围 的 起 始 值 。 

(3) 取 值 2: 表示 范围 的 终止 值 。 

例 9.6 查询 tb login 表 中 id 值 在 $-7 之 间 的 数据 , 查询 语句 如 下 。( 实 例 位 置 : 光盘 \TMsl\9\9.6) 

SELECT * FROM tb_login WHERE id BETWEEN 5 AND 7; 

查询 结果 如 图 9.5 所 示 。 








图 9.5 ”使 用 关键 字 BETWEEN AND 查询 
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如 果 要 查询 tb login 表 中 id 值 不 在 5~7 之 间 的 数据 ， 则 可 以 通过 NOT BETWEEN AND 来 完成 。 
其 查询 语句 如 下 。 
SELECT* FROM tb_login WHERE id NOT BETWEEN 5 AND 7; 





9.2.6 带 LIKE 的 字符 匹配 查询 


LIKE 属于 较 常用 的 比较 运算 符 , 通过 它 可 以 实现 模糊 查询 。 它 有 两 种 通配符 :“%” 和 下 划 线 “_”。 
(1) “%” 可 以 匹配 一 个 或 多 个 字符 ,可 以 代表 任意 长 度 的 字符 串 ， 长 度 可 以 为 0。 例 如，“ 明 % 
技 ” 表 示 以 “ 明 ” 开 头 ， 以 “ 技 ” 的 任意 长 度 的 字符 串 。 该 字符 串 可 以 代表 “明日 科技 ”“ 明 日 
编程 科技 ”“ 明 日 图 书 科技 ”等 字符 串 。 
(2) “_” 只 匹配 一 个 字符 。 例 如 ，m n 表示 以 m 开头 ， 以 n 结尾 的 3 个 字符 。 中 间 的 “ ”可 
以 代表 任意 一 个 字符 。 
he 
说 明 
字符 串 “p” 和 “入 ”都 算 作 一 个 字符 ， 在 这 点 上 英文 字母 和 中 文 是 没有 区 别 的 。 









例 9.7 查询 tb_login 表 中 user 字段 中 包含 mr 字符 的 数据 ， 查 询 语 句 如 下 。 (实例 位 置 ， 光盘 
VIMsI\9\9.7) 
select * from tb_login where user like '%mr%'; 


查询 结果 如 图 9.6 所 示 。 





et (0.00 sec> 


图 9.6 模糊 查询 


9.2.7 用 关键 字 IS NULL 查询 空 值 


关键 字 IS NULL 可 以 用 来 判断 字段 的 值 是 否 为 空 值 (NULL) 。 如 果 字 段 的 值 是 空 值 ， 则 满足 查 
询 条 件 ， 该 记录 将 被 查询 出 来 。 如 果 字 段 的 值 不 是 空 值 ， 则 不 满足 查询 条 件 。 其 语法 格式 如 下 。 

IS [INOT] NULL 

其 中 ，“NOT” 是 可 选项 ， 表 示 字 段 不 是 空 值 时 满足 条 件 。 

例 9.8 下 面 使 用 关键 字 IS NULL 查询 db_database09 数据 库 的 tb_book 表 中 name 字段 的 值 为 空 
的 记录 ， 查 询 语句 如 下 。《〔 实 例 位 置 : 光盘 \TMNsI\9\9.8) 

SELECT books,row FROM tb_book WHERE row IS NULL; 
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查询 结果 如 图 9.7 所 示 。 





图 9.7 查询 数据 表 tb_book 中 row 字段 值 为 空 的 记录 





9.2.8” 带 关键 字 AND 的 多 条 件 查询 


关键 字 AND 可 以 用 来 联合 多 个 条 件 进 行 查 询 。 使 用 关键 字 AND 时 ， 只 有 满足 所 有 查询 条 件 
的 记录 会 被 查询 出 来 。 如 果 不 满足 这 些 查询 条 件 的 其 中 一 个 ， 这 样 的 记录 将 被 排除 掉 。 关 键 字 AND 的 
语法 格式 如 下 。 

select * from 数据 表 名 where 条 件 1 and 条 件 2 [AND 条 件 表达 式 n]; 

关键 字 AND 连接 两 个 条 件 表达 式 ， 可 以 同时 使 用 多 个 关键 字 AND 来 连接 多 个 条 件 表达 式 。 

例 9.9 下 面 查询 数据 表 tb_login 中 user 字段 值 为 mr, 并 且 section 字段 值 为 PHP 的 记录 , 查询 语 
句 如 下 。《〔 实 例 位 置 ， 光盘 \TMNshv9\9.9) 

select * from tb_login where user='mr and section='php'; 

查询 结果 如 图 9.8 所 示 。 


nysql> 








图 9.8 ”使 用 关键 字 AND 实现 多 条 件 查询 


9.2.9 带 关 键 字 OR 的 多 条 件 查 询 


关键 字 OR 也 可 以 用 来 联合 多 个 条 件 进行 查询 ,但 是 与 关键 字 AND 不 同 ， 关键 字 OR 只 要 满足 查 
询 条 件 中 的 一 个 ， 那 么 此 记录 就 会 被 查询 出 来 ， 如 果 不 满 足 这 些 查询 条 件 中 的 任何 一 个 ， 这 样 的 记录 
将 被 排除 掉 。 关 键 字 OR 的 语法 格式 如 下 。 

select * from 数据 表 名 where 条 件 1 OR 条 件 2 […OR 条 件 表达 式 n]; 

关键 字 OR 可 以 用 来 连接 两 个 条 件 表达 式 。 而 且 ， 可 以 同时 使 用 多 个 关键 字 OR 连接 多 个 条 件 表 
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例 9.10 下 面 查询 tb_ login 表 中 section 字段 的 值 为 “PHP” 或 者 “程序 开发 ”的 记录 ， 查 询 语句 
如 下 。《〔 实 例 位 置 ， 光盘 \TMNsI\9\9.10) 
select * from tb_login where section='php' or section=' 程 序 开发 '; 
查询 结果 如 图 9.9 所 示 。 


n=" php” or section=’ 程 





图 99 使 用 关键 字 OR 实现 多 条 件 查询 
9.2.10 用 关键 字 DISTINCT 去 除 结果 中 的 重复 行 


使 用 关键 字 DISTINCT 可 以 去 除 查 询 结果 中 的 重复 记录 ， 语 法 格式 如 下 。 
select distinct 字段 名 from 表 名 ; 


例 9.11 下 面 使 用 关键 字 DISTINCT 去 除 tb_login 表 中 name 字段 中 的 重复 记录 
(实例 位 置 : 光盘 \TM'sI\9\9.11) 


select distinct name from tb_login; 


， 查 询 语句 如 下 。 


查询 结果 如 图 9.10 所 示 。 去 除 重复 记录 前 的 name 字段 值 如 图 9.11 所 示 。 


nane from th_login; 





(08.00 sec) 





sql 


图 9.10 使 用 关键 字 DISTINCT 去 除 结果 中 的 重复 行 图 9.11 





去 除 重复 记录 前 的 name 字段 值 


9.2.11 用 关键 字 ORDER BY 对 查询 结果 排序 


使 用 关键 字 ORDER BY 可 以 对 查询 的 结果 进行 升序 (ASC) 和 降序 (DESC) 排列 ， 在 默认 情况 
下 ，ORDER BY 按 升 序 输出 结果 。 如 果 要 按 降 序 排列 可 以 使 用 DESC 来 实现 。 语 法 格式 如 下 。 

ORDER BY 字段 名 [ASCIDESC]; 

(1) ASC 表示 按 升序 进行 排序 。 

(2) DESC 表示 按 降序 进行 排序 。 
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p94 
说明 

对 含有 NULL 值 的 列 进行 排序 时 ， 如 果 是 按 升序 排列 ，NULL 值 将 出 现在 最 前 面 ， 如 果 是 按 
降序 排列 ，NULL 值 将 出 现在 最 后 。 


例 9.12 查询 tb_login 表 中 的 所 有 信息 ,按照 id 序号 进行 降序 排列 ， 查 询 语句 如 下 。 (实例 位 置 : 
光盘 \TMsl\9\9.12) 

select* from tb_login order by id desc'; 

查询 结果 如 图 9.12 所 示 。 


nysql 





图 9.12 按 id 序 号 进行 降序 排列 


9.2.12 用 关键 字 GROUP BY 分 组 查询 


通过 关键 字 GROUP BY 可 以 将 数据 划分 到 不 同 的 组 中 ， 实 现 对 记录 进行 分 组 查询 。 在 查询 时 ， 所 
查询 的 列 必 须 包含 在 分 组 的 列 中 ， 目 的 是 使 查询 到 的 数据 没有 矛盾 。 

1. 使 用 关键 字 GROUP BY 来 分 组 

单独 使 用 关键 字 GROUP BY 查询 结果 只 显示 每 组 的 一 条 记录 。 

例 9.13 使 用 关键 字 GROUP BY 对 tb_book 表 中 talk 字段 进行 分 组 查询 ， 查 询 语句 如 下 。 (实例 
位 置 ， 光盘 \TMNsI\9\9.13) 

select id,books,talk from tb_book GROUP BY talk; 

查询 结果 如 图 9.13 所 示 。 





图 9.13 ”使 用 关键 字 GROUP BY 进行 分 组 查询 
为 了 使 分 组 更 加 直观 明了 ， 下 面 查 询 数据 表 tb_book 中 的 记录 ， 查 询 结果 如 图 9.14 所 示 。 
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fron th 





图 9.14 数据 表 tb_book 中 的 记录 

2. 关键 字 GROUP BY 与 GROUP_CONCAT() 函 数 一 起 使 用 

使 用 关键 字 GROUP BY 和 GROUP_ CONCAT0 函 数 查 询 ， 可 以 将 每 个 组 中 的 所 有 字段 值 都 显示 
出 来 。 

9.14 下 面 使 用 关键 字 GROUP BY 和 GROUP CONCATO 函 数 对 tb book 表 中 的 talk 字段 进行 

分 组 查询 ， 查 询 语 句 如 下 。 《实例 位 置 : 光盘 \TMNsl\9\9.14) 

select id,books,GROUP_CONCAT(talk) from tb_book GROUP BY talk; 

查询 结果 如 图 9.15 所 示 。 


book GROUP BY talk; 


(“8.80 sec 





图 9.15 使 用 关键 字 GROUP BY 与 GROUP_CONCATO 函 数 进行 分 组 查询 

3. 按 多 个 字段 进行 分 组 

使 用 关键 字 GROUP BY 也 可 以 按 多 个 字段 进行 分 组 。 

例 9.15 下 面 对 数 据 表 tb_book 表 中 的 user 字段 和 sort 字段 进行 分 组 ， 分 组 过 程 中 ， 先 按照 talk 
字段 进行 分 组 。 当 talk 字段 的 值 相等 时 ， 再 按照 sort 字段 进行 分 组 ， 查 询 语句 如 下 。 实例 位 置 ， 光 
盘 \TMNsl\9\9.15) 

select id,books,talk,user from tb_book GROUP BY user,talk; 


查询 结果 如 图 9.16 所 示 。 


r fron th_hook GROUP BY user,talk; 


JAUA ! mr 





图 9.16 使 用 关键 字 GROUP BY 实现 多 个 字段 分 组 
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9.2.13 ”用 关键 字 LIMIT 限制 查询 结果 的 数量 


查询 数据 时 ， 可 能 会 查询 出 很 多 的 记录 ， 而 用 户 需 要 的 记录 可 能 只 是 很 少 的 一 部 分 ， 这 样 就 需要 
来 限制 查询 结果 的 数量 。LIMIT 是 MySQL 中 的 一 个 特殊 关键 字 。 关 键 字 LIMIT 可 以 对 查询 结果 的 记 
录 条 数 进行 限定 ， 控 制 它 输出 的 行 数 。 下 面 通过 具体 实例 来 了 解 关 键 字 LIMIT 的 使 用 方法 
例 9.16 查询 数据 表 tb_login 


六 bh， 按照 id 编号 进行 升序 排列 ， 显 示 前 3 条 记录 ， 查 询 语 句 如 下 。 
(实例 位 置 ， 光盘 \TMNsI\9\9.16) 








select * from tb_login order by id asc limit 3; 
查询 结果 如 图 9.17 所 示 。 





图 9.17 使 用 关键 字 LIMIT 查询 指定 记录 数 
使 用 关键 字 LIMIT 还 可 以 从 查询 结果 的 中 间 部 分 取 值 。 
的 第 一 条 记录 的 编号 (在 查询 结果 中 ， 第 





首先 要 定义 两 个 参数 ， 
-个 纤 


一 个 结果 的 记录 编号 是 


参数 1 是 开始 读 取 
是 0， 而 不 是 1) ; 


参数 2 是 要 查询 记录 
的 个 数 。 
例 9.17 查询 tb_ login 表 中 ， 按 照 id 编 
句 如 下 。《〈 实 例 位 置 ， 光盘 \TNMNsIN9\9.17) 
select * from tb_login where id order by id asc limit 1,2; 


查询 结果 如 图 9.18 所 示 。 





号 进行 升序 排列 ， 从 编号 1 开始 ， 查 询 两 条 记录 ， 查 询 语 


Inysql> se 


login order by id asc linit 1.23 


lect * from th 


| section ! nane 
j! nrkj 
1 mrsoft 





图 9.18 ”使 用 关键 字 LIMIT 查询 指定 记录 


9.3 ”聚合 函数 查询 


聚合 函数 的 最 大 特点 是 它们 根据 一 组 数据 求 出 一 个 值 。 聚 合 函数 的 结果 值 只 根据 选 定 行 
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NULL 的 值 进行 计算 ，NULL 值 被 忽略 。 


9.3.1 COUNT() 函 数 




















COUNTO 函 数 ， 对 于 除 “*” 以 外 的 任何 参数 ， 返 回 所 选择 集合 中 非 NULL 值 的 行 的 数目 ， 对 于 
参数 “*”， 返 回 选择 集合 中 所 有 行 的 数目 ， 包 含 NULL 值 的 行 。 没有 WHERE 子 句 的 COUNT(*) 是 经 
过 内 部 优化 的 ， 能 够 快速 返回 表 中 所 有 的 记录 总 数 。 

例 9.18 下 面 使 用 count0 函 数 统计 数据 表 tb_login 中 的 记录 数 ， 查 询 语句 如 下 。 (实例 位 置 : 光 
盘 \TMNsl\9\9.18) 














select count(*) from tb_login; 
查询 结果 如 图 9.19 所 示 。 结 果 显示 ， 数 据 表 tb_login 中 共有 4 条 记录 。 


hysql> select countC*) from th_login; 


1 countCxe) } 


set 《698 sec) 





图 9.19 使 用 count0 函 数 统计 记录 数 


9.3.2 SUM() 函 数 


SUMO 函 数 可 以 求 出 表 中 某 个 字段 取 值 的 总 和 。 
例 9.19 使 用 SUMO 函 数 统计 数据 表 tb_book 中 图 书 的 访问 量 字 段 (row) 的 总 和 。 在 查询 前 ， 先 
来 查询 一 下 tb_book 表 中 row 字段 的 值 ， 结 果 如 图 9.20 所 示 。 (实例 位 置 ， 光盘 \TMNsl\9\9.19) 


select row from thb_hbookj 





图 9.20 tb_book 表 中 row 字段 的 值 
下 面 使 用 SUMO 函 数 来 查询 ， 查 询 语句 如 下 。 


select sum(row) from tb_book:; 


查询 结果 如 图 9.21 所 示 ， 显 示 row 字段 的 总 和 为 116。 
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ct sunCrow) from th_hook; 


row in set (8.82 sec) 





hysal> 


图 9.21 使 用 SUMO 函 数 查询 row 字段 值 的 总 和 


9.3.3 AVG() 函 数 


AVG0 函 数 可 以 求 出 表 中 某 个 字段 取 值 的 平均 值 。 

例 9.20 下 面 使 用 AVGO 函 数 求 数据 表 tb_book 中 row 字段 值 的 平均 值 ， 查 询 语句 如 下 。 《实例 
位 置 ; 光盘 \TMNsl\9\9.20) 

select AVG(row) from tb_book; 


查询 结果 如 图 9.22 所 示 。 


ysql> select AUGCrow) from th_book; 
! AUGCrow> ! 
29 ! 


row in set (8.80 sec) 


nysql> 


图 9.22 使 用 AVGO 函 数 求 row 字段 值 的 平均 值 





9.3.4 “MAX() 函 数 


MAXO 函 数 可 以 求 出 表 中 某 个 字段 取 值 的 最 大 值 。 

例 9.21 下 面 使 用 MAXO 函 数 查 询 数据 表 tb_book 中 row 字段 的 最 大 值 ， 查 询 语句 如 下 。 (实例 
位 置 : 光盘 \TMNsl\9\9.21) 

select MAX(row) from tb_book; 


查询 结果 如 图 9.23 所 示 。 


MAX Crow> from thb_book; 





图 9.23 ”使 用 MAXO 函 数 求 row 字段 的 最 大 值 
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下 面 来 看 一 下 数据 表 tb_ book 中 row 字段 的 所 有 值 ， 查 询 结果 如 图 9.24 所 示 。 结 果 显 示 row 字段 
中 最 大 值 为 935， 与 使 用 MAX 函数 查询 的 结果 一 致 。 





set (9.80 sec》 





图 9.24 数据 表 由 book 中 row 字段 的 所 有 值 
9.3.5_MIN() 函 数 


MINO 函 数 可 以 求 出 表 中 某 个 字段 取 值 的 最 小 值 。 
例 9.22 使 用 MINO 函 数 查 询 数据 表 tb_book 中 row 字段 的 最 小 值 ， 查 询 语句 如 下 。〔 实 例 位 置 : 
光盘 \TMNsI\9\9.22) 
select MIN(row) from tb_book; 
查询 结果 如 图 9.25 所 示 。 
MT pr 
i MINCrow> 1 
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ou in set 《8.98 sec? 





图 9.25 使 用 MINO 函 数 求 row 字段 的 最 小 值 


9.4 连接 查询 


连接 是 把 不 同 表 的 记录 连 到 一 起 的 最 普遍 的 方法 。 一 种 错误 的 观念 认为 由 于 MySQL 的 简单 性 和 
源 代 码 开放 性 , 使 它 不 擅长 连接 .MySQL 从 一 开始 就 能 够 很 好 地 支持 连接 , 现在 还 以 支持 标准 的 SQL92 
连接 语句 而 自豪 ， 这 种 连接 语句 可 以 以 多 种 高 级 方法 来 组 合 表 记 录 。 


9.4.1 内 连接 查询 


内 连接 是 最 普遍 的 连接 类 型 ， 而 且 是 最 匀称 的 ， 因 为 它们 要 求 构 成 连接 的 每 一 部 分 的 每 个 表 的 匹 
配 ， 不 匹配 的 行将 被 排除 。 
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内 连接 的 最 常见 的 例子 是 相等 连接 ， 也 就 是 连接 后 的 表 中 的 某 个 字段 与 每 个 表 中 的 都 相同 。 这 种 
情况 下 ， 最 后 的 结果 集 只 包含 参加 连接 的 表 中 与 指定 字段 相符 的 行 。 

例 9.23 下面 有 两 个 表 ，tb_login 用 户 信息 表 和 tb_book 图 书信 息 表 ， 先 来 分 别 看 下 各 表 的 数据 ， 
图 9.26 为 tb_login 表 的 数据 ， 图 9.27 为 tb_book 表 的 数据 。〔 实 例 位 置 ， 光盘 \TMNsI\9\9.23) 




















s from th_hook; 





图 9.27 数据 表 tb_book 
从 上 面 的 查询 结果 中 可 以 看 出 ， 在 两 个 表 中 存在 一 个 连接 一 一 user 字段 ， 它 在 两 个 表 中 是 等 同 的 ， 
tb_login 表 的 user 字段 与 tb_book 表 的 user 字段 相等 , 因此 可 以 创建 两 个 表 的 一 个 连接 。 查 询 语 句 如 下 。 
select name,books from tb_login,tb_book where tb_login.user=tb_book.user; 


查询 结果 如 图 9.28 所 示 。 





! books 





图 9.28 ”内 连接 查询 


148 


9.4.2 ”外 连接 查询 


与 内 连接 不 同 ， 外 连接 是 指使 用 OUTER JOIN 关键 字 将 两 个 表 连 接 起 来 。 外 连接 生成 的 结果 集 不 
仅 包 含 符合 连接 条 件 的 行 数据 ， 而 且 还 包括 左 表 ( 左 外 连接 时 的 表 ) 、 右 表 ( 右 外 连接 时 的 表 ) 或 两 
边 连 接 表 (全 外 连接 时 的 表 〉 中 所 有 的 数据 行 。 语 法 格式 如 下 。 

SELECT 字段 名 称 FROM 表 名 1 LEFTIRIGHT JOIN 表 名 2 ON 表 名 1. 字 段 名 1= 表 名 2. 属 性 名 2; 

外 连接 分 为 左 外 连接 (LEFT JOIN) 、 右 外 连接 (RIGHT JOIN) 和 全 外 连接 3 种 类 型 。 

1. 左 外 连接 

左 外 连接 (LEFT JOIN) 是 指 将 左 表 中 的 所 有 数据 分 别 与 右 表 中 的 每 条 数据 进行 连接 组 合 ， 返 回 
的 结果 除 内 连接 的 数据 外 ， 还 包括 左 表 中 不 符合 条 件 的 数据 ， 并 在 右 表 的 相应 列 中 添加 NULL 值 。 

例 9.24 下 面 使 用 左 外 连接 查询 tb_login 表 和 tb_book 表 ， 通 过 user 字段 进行 连接 ， 查 询 语句 如 

下 。《〈 实 例 位 置 : 光盘 \TMNsl\9\9.24) 
select section,tb_login.user,books,row from tb_login left join tb_book on tb_login.user=tb_book.user; 


查询 结果 如 图 9.29 所 示 。 








图 9.29 左 外 连接 查询 
结果 显示 ， 第 一 条 记录 的 books 和 row 字段 的 值 为 室 ， 这 是 因为 在 tb_book 表 中 并 不 存在 user 字 


段 为 mrkj 的 值 。 

2. 右 外 连接 

右 外 连接 (RIGHT JOIN) 是 指 将 右 表 中 的 所 有 数据 分 别 与 左 表 中 的 每 条 数据 进行 连接 组 合 ， 返 于 
的 结果 除 内 连接 的 数据 外 ， 还 包括 右 表 中 不 符合 条 件 的 数据 ， 并 在 左 表 的 相应 列 中 添加 NULL。 

例 9.25 下 面 使 用 右 外 连接 查询 tb_book 表 和 tb_login 表 ， 两 表 通 过 user 字段 连接 ， 查 询 语句 如 
下 。《〔 实 例 位 置 : 光盘 \TMNsI\9\9.25) 


select section,tb_book.user,books,row from tb_book right join tb_login on tb_book.user=tb_login.user; 
查询 结果 如 图 9.30 所 示 。 
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right join th_login on 





图 9.30 右 外 连接 查询 


9.4.3 复合 条 件 连接 查询 


在 连接 查询 时 ， 也 可 以 增加 其 他 的 限制 条 件 。 通 过 多 个 条 件 的 复合 查询 ， 可 以 使 查询 结果 更 加 


准确 。 
例 9.26 
(实例 位 置 ， 光 盘 \TMNsI\9\9.26) 


下 面 使 用 内 连接 查询 tb_book 表 和 tb_login 表 , 并且 tb_book 表 中 row 字段 值 必须 大 于 5， 
查询 语句 如 下 。 
select section,tb_book.user,books,row from tb_book,tb_login where tb_book.user=tb_login.user and row>5; 


login where th_hoo 


查询 结果 如 图 9.31 所 示 。 





复合 条 件 连 接 查 询 


图 9.31 


9.5 子 查询 


子 查询 就 是 SELECT 查询 是 另 一 个 查询 的 附属 。MySQL 4.1 可 以 说 套 多 个 查询 ， 在 外 面 一 层 的 查 
Pp 使 用 里 面 一 层 查 询 产生 的 结果 集 。 这 样 就 不 是 执行 两 个 (或 者 多 个 ) 独立 的 查询 ， 而 是 执行 包含 
外 向 上 移动 到 外 层 ( 主 ) 





询 
-个 (或 者 多 个 ) 子 查询 的 单独 查询 。 

当 过 到 这 样 的 多 层 查 询 时 , MySQL 从 最 内 层 的 查询 开始 , 然后 从 它 开始 向 
查询 ， 在 这 个 过 程 中 每 个 查询 产生 的 结果 集 都 被 赋 给 包围 它 的 父 查 询 ， 接 着 这 个 父 查询 被 执行 ， 它 的 





结果 也 被 指定 给 父 查 询 。 
除了 结果 集 经 常 由 包含 一 个 或 多 个 值 的 一 列 组 成 外 , 子 查询 和 常规 SELECT 查询 的 执行 方式 一 样 。 
子 查询 可 以 用 在 任何 可 以 使 用 表达 式 的 地 方 ， 它 必须 由 父 查询 包围 ,而且 ， 如 同 常规 的 SELECT 查询 ， 
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它 必须 包含 一 个 字段 列表 (这 是 一 个 单列 列表 ) 、 一 个 具有 一 个 或 者 多 个 表 名 字 的 FROM 子 句 以 及 可 
选 的 WHERE、HAVING 和 GROUP BY 子 句 。 


9.5.1 带 关键 字 IN 的 子 查询 








的 结果 集 是 值 








只 有 子 查询 返回 的 结果 列 包含 一 个 值 时 ， 比 较 运 算 符 才 适用 。 假 如 一 个 子 查询 返 
的 列表 ， 这 时 比较 运算 符 就 必须 用 关键 字 IN 代 蔡 。 

IN 运算 符 可 以 检测 结果 集中 是 否 存在 某 个 特定 的 值 ， 如 果 检 测 成 功 就 执行 外 部 的 查询 。 

例 9.27 下 面 查询 tb _ login 表 中 的 记录 , 但 user 字段 值 必须 在 tb_book 表 中 的 user 字段 中 出 现 过 ， 
查询 语句 如 下 。〔 实 例 位 置 ， 光盘 \TMNsN\9\9.27) 

select * from tb_login where user in(select user from tb_book); 

在 查询 前 ， 先 来 分 别 看 一 下 tb_login 和 tb_book 表 中 的 user 字段 值 ， 以 便 进行 对 比 ，tb_login 表 中 
的 user 字段 值 如 图 9.32 所 示 。tb_book 表 中 的 user 字段 值 如 图 9.33 所 示 。 

















Inysql> select user from th_hook; 


Imysql> select user from th_login; 


in set 《0.00 sec) rows in set (0.00 sec) 





图 9.32 tb_login 表 中 的 user 字段 值 图 9.33 ”tb_book 表 中 的 user 字段 值 





从 上 面 的 查询 结果 可 以 看 出 , 在 tb_book 表 的 user 字段 中 没有 出 现 mrkj 值 。 下 面 执行 带 关 键 字 IN 
的 子 查询 语句 ， 查 询 结 果 如 图 9.34 所 示 。 


er fron th_hook); 





图 9.34 使 用 关键 字 IN 实现 子 查询 
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查询 结果 只 查询 出 了 user 字段 值 为 和 mr 的 记录 ,因为 在 tb_book 表 的 user 字段 中 没有 出 现 mrkj 
的 值 。 
wa 
-说明 
关键 字 NOT IN 的 作用 与 关键 字 IN 刚好 相反 。 在 本 例 中 ， 如 果 将 IN 换 为 NOT IN， 则 查询 结 
果 将 会 只 显示 一 条 user 字段 值 为 mrkj 的 记录 。 





9.5.2 ” 带 比 较 运算 符 的 子 查询 


于 查询 可 以 使 用 比较 运算 符 ， 包 括 =、!=、>、>=、< 和 <= 等 。 比 较 运算 符 在 子 查询 时 使 用 非常 
广泛 。 

例 9.28 下 面 查询 图 书 访问 量 为 “优秀 ”的 图 书 , 在 tb_row 表 中 将 图 书 访问 量 按 访 问 数 划 分 等 级 ， 
如 图 9.35 所 示 。 《实例 位 置 : 光盘 \TMNsI\9\9.28) 

从 结果 中 看 出 ， 当 访问 量 大 于 等 于 90 时 即 为 “优秀 ”， 下 面 再 来 查询 tb_book 图 书信 息 表 中 row 
字段 的 值 ， 如 图 9.36 所 示 。 





图 9.35 查询 由 row 表 中 的 数据 图 9.36 查询 tb_book 表 中 row 字段 的 值 
结果 显示 , 第 27 条 记录 的 访问 量 大 于 90。 下面 使 用 比较 运算 符 的 子 查询 方式 来 查询 访问 量 为 优秀 
的 图 书信 息 ， 查 询 语句 如 下 。 
select id,books,row from tb_book where row>=(select row from tb_row where id=1); 


查询 结果 如 图 9.37 所 示 。 








from th_hook where row>=Cselect row fron th_row wherd 





图 9.37 使 用 比较 运算 符 的 子 查询 方式 来 查询 访问 量 为 “优秀 ”的 图 书信 息 
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9.5.3 ”和 带 关键 字 EXISTS 的 子 查询 


使 用 关键 字 EXISTS 时 ， 内 层 查 询 语 句 不 返回 查询 的 记录 。 而 是 返回 一 个 真 假 值 。 如 果 内 层 查询 
语句 查询 到 满足 条 件 的 记录 ， 就 返回 一 个 真 值 (tue) ， 和 否则 将 返回 一 个 假 值 (false) 。 当 返回 的 值 为 
true 时 ， 外 层 查询 语句 将 进行 查询 ， 当 返回 的 值 为 false 时 ， 外 层 查询 语句 不 进行 查询 或 者 查询 不 出 任 
何 记 录 。 

例 9.29 下 面 使 用 子 查询 查询 tb_book 表 中 是 否 存在 id 值 为 27 的 记录 ， 如 果 存 在 则 查询 tb_row 
表 中 的 记录 ， 如 果 不 存在 则 不 执行 外 层 查询 ， 查 询 语 句 如 下 。 (实例 位 置 : 光盘 \TMNsl\9\9.29) 

select * from tb_row where exists (select * from tb_book where id=27); 


查询 结果 如 图 9.38 所 示 。 









































> select * from th_row where existsCselect * fron th_hook where id 





图 9.38 ”使 用 关键 字 EXISTS 的 子 查询 

因为 子 查询 tb_book 表 中 存在 id 值 为 27 的 记录 ， 即 返回 值 为 真 ， 外 层 查 询 接 收 到 真 值 后 ， 开 始 执 
行 查询 。 

当 关 键 字 EXISTS 与 其 他 查询 条 件 一 起 使 用 时 ， 需 要 使 用 AND 或 者 OR 来 连接 表达 式 与 EXISTS 
关键 字 。 

例 9.30 如 果 tb_row 表 中 存在 name 值 为 “优秀 ”的 记录 ， 则 查询 tb_book 表 中 row 字段 大 于 等 
于 90 的 记录 ， 查 询 语句 如 下 。 《实例 位 置 : 光盘 \TMNsl\9\9.30) 

select id,books,row from tb_book where row>=90 and exists(select * from tb_row where name=' 优 秀 '); 

查询 结果 如 图 9.39 所 示 。 

nysql> select iad Pow fron th_hook where row>=98 and existsCselect x* from ¢ 


b_row where name 








到 








9.39 ”使 用 关键 字 EXISTS 查询 tb_book 表 中 row 字段 大 于 等 于 90 的 记录 
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人 
说 明 
与 关键 字 EXISTS 刚好 相反 ， 使 用 关键 字 NOT EXISTS 时 ， 当 返回 的 值 是 true 时 ， 外 层 查询 
语句 不 执行 查询 ; 当 返 回 值 是 false 时 ， 外 层 查 询 语句 将 执行 查询 。 


9.5.4” 带 关键 字 ANY 的 子 查询 





关键 字 ANY 表示 满足 其 中 任意 一 个 条 件 。 使 用 关键 字 ANY 时 ， 只 要 满足 内 层 查 询 语句 返回 的 结 
果 中 的 任意 一 个 ， 就 可 以 通过 该 条 件 来 执行 外 层 查 询 语句 。 
例 9.31 查询 tb_book 表 中 row 字段 的 值 小 于 tb_row 表 中 row 字段 最 小 值 的 记录 ， 首 先 查 询 出 
tb_row 表 中 row 字段 的 值 ， 然 后 使 用 关键 字 ANY (“<ANY” 表 示 小 于 所 有 值 ) 判断 ， 查 询 语 句 如 下 。 
(实例 位 置 : 光盘 \TMNsI\9\9.31) 


select books,row from tb_book where row<ANY(select row from tb_row); 














查询 结果 如 图 9.40 所 示 。 





图 9.40 使 用 关键 字 ANY 实现 子 查询 


为 了 使 结果 更 加 直观 ， 下 面 分 别 查询 tb_book 表 和 tb_row 表 中 的 row 字段 值 ， 查 询 结果 如 图 9.41 
和 图 9.42 所 示 。 


nysql> select row fron th_hook; 


对 ! 
1 NULL ! 





6 rows in set 9.86 sec) 





图 9.41 tb_book 表 中 row 字 段 的 值 图 9.42 ”tb_row 表 中 row 字 段 的 值 
结果 显示 ，tb_row 表 中 row 字段 的 最 小 值 为 50， 在 tb_book 表 中 row 字段 小 于 50 的 记录 有 3 条 ， 
与 带 关 键 字 ANY 的 子 查询 结果 相同 。 





9.5.5” 带 关键 字 ALL 的 子 查询 





关键 字 ALL 表示 满足 所 有 条 件 。 使 用 关键 字 ALL 时 ， 只 有 满足 内 层 查询 语句 返回 的 所 有 结果 ， 
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才 可 以 执行 外 层 查询 语句 。 
例 9.32 查询 tb_book 表 中 row 字段 的 值 大 于 tb_row 表 中 row 字段 最 大 值 的 记录 ,首先 使 用 子 查 


询 ， 查 询 出 tb_row 表 中 row 字段 的 值 ， 然 后 使 用 ALL 关键 字 〈〔“>=ALL” 表 示 大 于 等 于 所 有 值 ) 判 
断 ， 查 询 语句 如 下 。 《实例 位 置 : 光盘 \TMNsl\9\9.32) 


select books,row from tb_book where row>=ALL(select row from tb_row); 
查询 结果 如 图 9.43 所 示 。 








row from th_hook where row> 


alllselect row fron th_row); 





图 9.43 ”使 用 关键 字 ALL 实现 子 查询 


为 了 使 结果 更 加 直观 ， 下 面 分 别 查询 tb_book 表 和 tb _ row 表 中 的 row 


字段 值 ， 查 询 结果 如 图 9.44 
和 图 9.45 所 示 。 


> select row from th_hook; 


> select row from th_row; 


in set (0.85 sec) 


5 rows in set C0.86 sec) 





图 9.44 tb_book 表 中 row 字段 的 值 图 9.45 tb_row 表 中 row 字段 的 值 


s，tb_row 表 中 row 字段 的 最 大 值 为 90, 在 tb_book 表 中 row 字段 大 于 90 的 只 有 第 2 条 记 
录 95， 与 带 关 键 字 ALL 的 子 查询 结果 相同 。 
{说明 

关键 字 ANY 和 关键 字 ALL 的 使 用 方式 是 一 样 的 ,但 是 这 两 者 有 很 大 的 区 别 .使 用 关键 字 ANY 
时 ， 只 要 满足 内 层 查询 语句 返回 的 结果 中 的 任何 一 个 ,就 可 以 通过 该 条 件 来 执行 外 层 查询 语句 ; 而 
关键 字 ALL 则 需要 满足 内 层 查 询 语 名 返回 的 所 有 结果 ， 才 可 以 执行 外 层 查 询 语 句 。 


结果 最 志 


9.6 合并 查询 结果 





合并 查询 结果 是 将 多 个 SELECT 语句 的 查询 结果 
SELECT 语句 查询 出 来 的 结果 合并 起 来 显示 





合并 到 一 起 。 因 为 某 种 情况 下 ， 需 要 将 几 个 
。 合 并 查询 结果 使 用 关键 字 UNION 和 UNION ALL。 关 键 
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字 UNION 是 将 所 有 的 查询 结果 合并 到 一 起 , 然后 去 除 相 同 记 录 ; 而 关键 字 UNION ALL 则 只 是 简单 地 
将 结果 合并 到 一 起 ， 下 面 分 别 介绍 这 两 种 合并 方法 。 








1. UNION 

例 9.33 下 面 查 询 tb_book 表 和 tb login 表 中 的 user 字段 ， 并 使 用 UNION 关键 字 合 并 查询 结果 。 
在 执行 查询 操作 前 , 先 来 看 一 下 tb_book 表 和 tb_login 表 中 user 字段 的 值 , 查询 结果 如 图 9.46 和 图 9.47 
所 示 。【〔 实 例 位 置 : 光盘 \TMNsI\9\9.33) 


elect user from th_hook; 








> select user from th_login; 


set (0.83 sec》 


in set 0.86 sec》 





图 9.46 tb_book 表 中 user 字段 的 值 图 9.47 tb login 表 中 user 字段 的 值 
结果 显示 ,在 tb_book 表 中 user 字段 的 值 有 两 种 ， 分 别 为 mr 和 lx， 而 tb_login 表 中 user 字段 的 值 
有 三 种 。 下 面 使 用 关键 字 UNION 合并 两 个 表 的 查询 结果 ， 查 询 语句 如 下 。 


select user from tb_book 
UNION 
select user from tb_login; 


查询 结果 如 图 9.48 所 示 。 结 果 显示 ， 合 并 后 将 所 有 结果 合并 到 了 一 起 ， 并 去 除了 重复 值 。 
select user from th_hook 


> select user from th_login; 


+ 
in set 《0@.88 sec) 





图 9.48 ”使 用 关键 字 UNION 合并 查询 结果 
2. UNION ALL 
查询 tb_book 表 和 tb_login 表 中 的 user 字段 , 并 使 用 关键 字 UNION ALL 合并 查询 结果 ,查询 语 和 名 
如 下 。 


select user from tb_book 
UNION ALL 
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select user from tb_login; 
查询 结果 如 图 9.49 所 示 。tb_book 表 和 tb login 表 的 记录 请 参见 例 9.33。 


"于 rom th_hook 


rom th_login; 


9 rows in set 《0.00 sec) 
Inysql> 


图 9.49 使 用 关键 字 UNION ALL 合并 查询 结果 





9.7 定义 表 和 字段 的 别名 


在 查询 时 ， 可 以 为 表 和 字段 取 一 个 别名 ， 这 个 别名 可 以 代替 其 指定 的 表 和 字段 。 为 字段 和 表 取 别 
名 ， 能 够 使 查询 更 加 方便 ， 而 且 可 以 使 查询 结果 以 更 加 合理 的 方式 显示 。 


9.7.1 为 表 取 别名 


当 表 的 名 称 特别 长 时 ， 在 查询 中 直接 使 用 表 名 很 不 方便 ， 这 时 可 以 为 表 取 一 个 贴切 的 别名 。 

例 9.34 下 面 为 tb_program 表 取 别名 为 p， 然 后 查询 tb_program 表 中 talk 字段 值 为 php 的 记录 ， 
查询 语句 如 下 。《〈 实 例 位置 : 光盘 \TMNsiv9\9.34) 

select * from tb_program p where ptalk='PHP': 

tb_program p 表示 tb_program 表 的 别名 为 p; p.talk 表示 tb_program 表 中 的 talk 字段 。 查 询 结果 如 
图 9.50 所 示 。 


an p where p-talk=’PHP’; 











9.50 ”为 表 取 别名 
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9.7.2 为 字段 取 别 名 


当 查 询 数 据 时 ，MySQL 会 显示 每 个 输出 列 的 名 词 。 默 认 情况 下 ， 显 示 的 列 名 是 创建 表 时 定义 的 列 
名 。 同 样 可 以 为 这 个 列 取 一 个 别名 。 
MySQL 中 为 字段 取 别 名 的 基本 形式 如 下 。 





字段 名 [AS] 














别名 





例 9.35 下 面 为 tb login 表 中 的 section 和 name 字段 分 别 取 别名 为 login_section 和 login_name， 


SQL 代码 如 下 。 


(实例 位 置 ， 光盘 \TMNsl\9\9.3S) 


select section AS login_section,name AS login_name from tb_login; 


查询 结果 如 图 9.51 所 示 。 





图 9.51 为 字段 取 别 名 


9.8 使 用 正则 表达 式 查询 


正则 表达 式 是 用 某 种 模式 去 匹配 一 类 字符 串 的 一 个 方式 。 正 则 表达 式 的 查询 能 力 比 通 配 字符 的 查 


询 能 力 更 强大 ， 


而 且 更 加 灵活 。 下 面 详细 讲解 如 何 使 用 正则 表达 式 来 查询 。 


在 MySQL 中， 使 用 关键 字 REGEXP 来 匹配 查询 正则 表达 式 ， 其 基本 形式 如 下 。 

字段 名 REGEXP ' 匹 配方 式 ' 

(1) 字段 名 : 表示 需要 查询 的 字段 名 称 。 

(2) 匹配 方式 ， 表示 以 哪 种 方式 来 进行 匹配 查询 。 其 支持 的 模式 匹配 字符 如 表 9.2 所 示 。 


模式 字符 





表 9.2 正则 表达 式 的 模式 字符 
4 应 用 举例 
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用 “^” 表 达 式 查询 也 _book 表 中 books 字段 以 字母 php 开头 的 
匹配 以 特定 字符 或 字符 串 开 头 的 使 用 ee 询 也 _book 表 中 books 字段 以 字母 php 开头 的 
记录 记录 ， 语 句 如 下 : 








select books from tb book where books REGEXP “php'; 


续 表 
应 用 举例 
使 用 “$” 表 达 式 查询 tb_book 表 中 books 字段 以 “模块 ”结尾 
的 记录 ， 语 句 如 下 : 
select books from tb_book where books REGEXP "模块 $'; 


匹配 以 特定 字符 或 字符 串 结尾 的 
记录 





[字符 集合 ] 





使 用 “.” 表 达 式 来 查询 tb_book 表 中 books 字段 中 包含 P 字 符 的 
记录 ， 语 句 如 下 : 

select books from tb _ book where books REGEXP 'P.': 

使 用 “ 口 ”表达 式 来 查询 tb_book 表 中 books 字段 中 包含 PCA 字 
符 的 记录 ， 语 句 如 下 : 


匹配 字符 串 的 任意 一 个 字符 ， 包 
括 回 车 和 换行 符 














匹配 “字符 集合 ”中 的 任意 一 个 











了 select books from tb_book where books REGEXP '[PCAT': 
3 ~ Iy 了 证 
匹配 除 “ 字 符 集合 ”以 外 的 任意 查询 t program 表 中 talk 字段 值 中 包含 c~z 字母 以 外 的 记录 , 语 
[字符 集合 ] | 一 个 字符 句 如 下 : 
select talk from tb_program Where talk regexp '[^c-z]'; 
查询 tb_books 表 中 books 字段 中 包含 php、c 或 者 java 字符 中 任 
匹 、 意 一 
S1|S2|S3 二 人 作 的 全 让 意 一 个 字符 的 记录 ， 语 句 如 下 : 
select books from tb_books where books regexp 'phplcljava'; 
使 用 “* ”表达 式 查 询 tb_book 表 中 books 字段 中 A 字符 前 出 现 
匹配 多 个 该 条 符 ， Se 
， 报时 之 朋 四条， 外 | 过 字符 的 记录 ,语句 如 下 
select books from tb book where books regexp J*A'; 
2 使 用 “+” 表 达 式 来 查询 tb_book 表 中 books 字段 中 A 字符 前 面 
+ ep td 包 | 至 少 出 现 过 一 个 字符， 语句 如 下 : 
select books from tb_book where books regexp J+A'; 
使 用 {N} 表 达 式 查询 tb_book 表 中 books 字段 中 连续 出 现 3 次 a 
字符 串 {N} 匹配 字符 串 出 现 N 次 字符 的 记录 ， 语 句 如 下 : 
select books from tb_book where books regexp 'a{3}'; 
re 使 用 {MN} 表 达 式 查询 tb_book 表 中 books 字 段 中 最 少 出 现 两 次 ， 
字符 趾 {MN} | 亚 配 字符 审 出 现 至 少 M 次 , 最 多 | 最 多 出 现 4 次 a 字符 的 记录 ,语句 如 下 : 





N 次 


select books from tb_book where books regexp 'a{2.4}': 


这 里 的 正则 表达 式 与 Java 语言 、PHP 语言 等 编程 语言 中 的 正则 表达 式 基本 一 致 。 





9.8.1 匹配 指定 字符 中 的 任意 一 个 





使 用 方 括号 〈[]) 可 以 将 需要 查询 字符 组 成 一 个 字符 集 。 只 要 记录 中 包含 方 括号 中 的 任意 字符 ， 该 
记录 将 会 被 查询 出 来 。 例 如 ， 通 过 “[abc]” 可 以 查询 包含 a、b 和 c3 个 字母 中 任何 一 个 的 记录 。 

例 9.36 ”下面 从 info 表 name 字段 中 查询 包含 c、e 和 3 个 字母 中 任意 一 个 的 记录 。SQL 代码 如 
下 。 《实例 位 置 : 光盘 \TMslv9\9.36) 
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SELECT * FROM info WHERE name REGEXP [ceo]': 
代码 执行 结果 如 图 9.52 所 示 。 


select x* from info where name regexp ’[ceo]’; 


1 Bread 
i ahbcl2 } 


s in set CG.81 sec》 





图 9.52 匹配 指定 字符 中 的 任意 一 个 
9.8.2 使 用 “*” 和 “+” 来 匹配 多 个 字符 


正则 表达 式 中 ，“*” 和 “+” 都 可 以 匹配 多 个 该 符号 之 前 的 字符 。 但是，“+” 至 少 表示 一 个 字符 ， 
而 “*” 可 以 表示 0 个 字符 

例 9.37 下 面 从 info 表 name 字段 中 查询 字母 “c” 之 前 出 现 过 “a” 的 记录 。SQL 代码 如 下 : 〈 实 
例 位 置 ， 光 盘 \TM'Nsl\9\9.37) 

SELECT * FROM info WHERE name REGEXP 'a*c'; 


代码 执行 结果 如 图 9.53 所 示 。 





select x* from info where name regexp ’axc’; 


in set 《0.80 sec》> 





图 9.53 使 用 “*” 来 匹配 多 个 字符 








查询 结果 显示 ，Aric、Eric 和 Lucy 中 的 字母 之 前 并 没有 a。 因为 “*” 可 以 表示 0 个， 所 以 “a*c” 


表示 字母 之 前 有 0 个 或 者 多 个 a 出 现 。 上 述 的 情况 都 是 属于 前 面 出 现 过 0 个 的 情况 。 如 果 使 用 “+”， 
其 SQL 代码 如 下 : 


SELECT * FROM info WHERE name REGEXP ‘a+c’; 
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代码 执行 结果 如 图 9.54 所 示 。 


1> select * from info where name regexp ’a+c’s 


-B8 sec)> 





图 9.54 使 用 “+” 来 匹配 多 个 字符 






查询 结果 只 有 一 条 。 只 有 Jack 是 刚好 字母 前 面 出 现 了 a。 因 为 atc 表示 字母 c 前 面 至 少 有 一 个 


字母 a。 


9.8.3 ”匹配 以 指定 的 字符 开头 和 结束 的 记录 


正则 表达 式 中 ，^ 表 示 字 符 串 的 开始 位 置 ，$ 表 示 字 符 串 的 结束 位 置 。 下 面 将 通过 一 个 具体 的 实例 

演示 如 何 匹配 以 指定 的 字符 开头 和 结束 的 记录 
例 9.38 在 学 生成 绩 信 息 表 computer_stu 中 查找 姓名 Cname) 字段 中 以 工 开头 、 以 y 结 束 的 ， 中 

间 包 含 两 个 字符 的 学 生 的 成 绩 信息 ， 运 行 结果 如 图 9.55 所 示 。 (实例 位 置 : 光盘 \TMNsl\9\9.38) 

e db_databaseQ@6; 


e changed 
CT # FROM computer_stu WHERE name REGEXP 








Cs 





Inysql> 


图 9.55 使 用 正则 表达 式 查询 学 生成 绩 信息 
要 实现 查询 姓名 (name) 字段 中 以 工 开头 、 以 7 结 ” 的 ， 中 间 包 含 两 个 写 
过 正则 表达 式 查询 来 实现 。 其 中 正则 表达 式 中 ，^ 表 示 字 - 符 审 的 开始 位 置 ， $ 表 示 
表示 除 “\n” 以 外 的 任何 单个 字符 ， 具 体 代 码 如 下 。 
SELECT * FROM computer_stu WHERE name REGEXP '^L..y$'; 





:成 绩 ， 可 以 通 
结束 位 置 ，. 








2 和 处 结 





本 章 对 MySQL 数据 库 常 见 的 查询 方法 进行 了 详细 讲解 ， 并 通过 大 量 的 举例 说 明 ， 帮 助 读者 更 好 


地 理解 所 学 知识 的 用 法 。 在 阅读 本 章 时 ， 读 者 应 该 重点 掌握 多 条 件 查询 、 连 接 查 询 、 子 查询 和 查询 结 
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果 排 序 。 本 章 学 习 的 难点 是 使 用 正则 表达 式 来 查询 。 正 则 表达 式 的 功能 很 强大 ， 使 用 起 来 很 灵活 。 希 
望 读 者 能 够 阅读 有 关 正 则 表达 式 的 相关 知识 ， 能 过 对 正则 表达 式 了 解 得 更 加 透彻 。 


9.10 ”实践 与 练习 


1. 实现 从 computer stu 表 中 查询 获得 一 等 奖学金 的 学 生 的 学 号 、 姓 名 和 分 数 。 各 个 等 级 的 奖学金 
的 最 低 分 存储 在 score 表 中 。 (答案 位 置 : 光盘 \TMNsh\9\9.39) 
2. 实现 匹配 computer_stu 表 中 姓名 字段 中 以 J 开头 的 记录 。〔 答 案 位 置 ， 光盘 \TMsl\9\9.40) 
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ss 


常用 函数 


( 馈 ( 视频 讲解 ，36 分 钟 ) 


My5QL 数据 库 中 提供 了 很 丰富 的 函数 。My5QL 函数 包括 数学 函数 、 字符 串 范 
数 、 日 期 和 时 间 函 数 、 条 件 判断 函数 、 系 统 信息 函数 、 加 窗 函 数 、 格 式 化 函数 等 。 
函数 的 执行 速度 非常 快 ， 可 以 提高 My5QL 的 处 理 速度 ， 简 化 用 户 的 操作 。 本 章 将 
详细 介绍 MySQL 函 教 的 相关 知识 。 

通过 阅读 本 章 ， 读 者 可 以 : 
了 解 MySQL 国 数 
了 解 MySQL 数学 国 数 的 使 用 方法 
掌 查 MySQL 字符 捉 困 数 的 使 用 方法 
掌握 MySQL 日 期 和 时 间 国 数 的 使 用 方法 
掌握 MySQL 条 件 判断 印 数 的 应 用 方法 
党 握 系 统 信息 国 数 和 加 密 国 数 的 使 用 方法 


于 于 于 至 至 至 
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10.1 MySQL 函数 


MySQL 函数 是 MySQL 数据 库 提供 的 内 置 函 数 。 这 些 内 置 函 数 可 以 帮助 用 户 更 加 方便 地 处 理 表 中 
的 数据 。 本 节 将 简单 地 介绍 MySQL 中 包含 哪些 类 别 的 函数 ， 以 及 这 些 函 数 的 使 用 范围 和 作用 ， 如 表 
10.1 所 示 。 
表 10.1 MySQL 内 置 函数 类 别 及 作用 
函数 作 用 
数学 函数 用 于 处 理 数字 。 这 类 函数 包括 绝对 值 函数 、 正 弦 函 数 、 余 弦 函 数 和 获取 随机 数 函数 等 
字符 串 函 数 用 于 处 理 字符 串 。 其 中 包括 字符 串 连 接 函数 、 字 符 串 比较 函数 、 字 符 串 中 字母 大 小 写 转换 函数 等 


用 于 处 理 日 期 和 时 间 。 其 中 包括 获取 当前 时 间 的 函数 、 获 取 当前 日 期 的 函数 、 返 回 年 份 的 函数 和 
返回 日 期 的 函数 等 


条 件 判断 函数 用 于 在 SQL 语句 中 控制 条 件 选 择 。 其 中 包括 正 语句 、CASE 语句 和 WHEN 语句 等 
系统 信息 函数 用 于 获取 MySQL 数据 库 的 系统 信息 。 其 中 包括 获取 数据 库 名 的 函数 、 获 取 当 前 用 户 的 函数 和 获 








日 期 和 时 间 函 数 





取 数 据 库 版 本 的 函数 等 
加 密 函数 用 于 对 字符 串 进行 加 密 解密 。 其 中 包括 字符 串 加 密 函数 和 字符 串 解密 函数 等 
其 他 函数 包括 格式 化 函数 和 锁 函 数 等 


MySQL 的 内 置 函 数 不 但 可 以 在 SELECT 查询 语句 中 应 用 ， 同 样 也 可 以 在 INSERT、UPDATE 和 
DELECT 等 语句 中 应 用 。 例 如 ， 在 INSERT 语句 中 ， 应 用 日 期 时 间 函 数 获 取 系统 的 当前 时 间 ， 并 且 将 
其 添加 到 数据 表 中 。MySQL 内 置 函数 可 以 对 表 中 数据 进行 相应 的 处 理 , 以 便 得 到 用 户 希 望 得 到 的 数据 。 
有 了 这 些 内 置 函 数 可 以 使 MySQL 数据 库 的 功能 更 加 强大 。 下面 将 对 MySQL 的 内 置 函 数 逐 一 进行 详细 
介绍 。 


10.2 数学 函数 


数学 函数 是 MySQL 中 常用 的 一 类 函数 ， 主 要 用 于 处 理 数 字 ， 包 括 整 型 和 浮 点 数 等 。MySQL 中 内 
置 的 数学 函数 及 其 作用 如 表 10.2 所 示 。 
表 10.2 MySQL 的 数学 函数 





























函数 作 用 
ABsG | 返回 x 的 绝对 值 

CEILGO.CEILINGJ 。 | 返回 不 小 于 x 的 最 小 整数 值 

FLOOR() | 返回 不 大 于 x 的 最 大 整数 值 

RANDO | 返回 0~1 的 随机 数 

RANDG 返回 0~1 的 随机 数 ，x 值 相同 时 返回 的 随机 数 相同 
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续 表 
函数 作 用 
SIGN(x) 返回 参数 作为 -1、0 或 1 的 符号 ， 该 符号 取决 于 x 的 值 为 负 、 零 或 正 
PIO 返回 圆周 率 的 值 。 默 认 的 显示 小 数位 数 是 7 位 ， 然 而 MySQL 内 部 会 使 用 完全 双 精 度 值 
TRUNCATE(x.y) 返回 数值 x 保留 到 小 数 点 后 y 位 的 值 
ROUND(x) 返回 离 x 最 近 的 整数 
ROUND(x.y) 保留 x 小数 点 后 y 位 的 值 ， 但 截断 时 要 进行 四 舍 五 入 
POW(xX.y).POWER(x.y) | 返回 x 的 y 乘 方 的 结果 值 
SQRT(x) 返回 非 负数 x 的 二 次 方 根 
EXP(x) 返回 e 的 x 乘 方 后 的 值 ( 自 然 对 数 的 底 ) 
MOD(x.y) 返回 x 除 以 y 以 后 的 余数 
LOG(x) 返回 x 的 基数 为 2 的 对 数 
LOG10(x) 返回 x 的 基数 为 10 的 对 数 
RADIANS(x) 将 角度 转换 为 弧度 
DEGREES(x) 返回 参数 x， 该 参数 由 弧度 转化 为 度 
SIN(x) 返回 x 的 正弦 ， 其 中 x 在 弧度 中 被 给 定 
ASIN(x) 返回 x 的 反正 弦 ， 即 正弦 为 x 的 值 。 若 x 不 在 -1~1 的 范围 之 内 ， 则 返回 NULL 
COS(x) 返回 x 的 余弦 ， 其 中 x 在 弧度 上 已 知 
ACOS(x) 返回 x 的 反 余弦 ， 即 余弦 是 x 的 值 。 若 x 不 在 -1~1 的 范围 之 内 ， 则 返回 NULL 
TAN(x) 返回 x 的 反正 切 ， 即 正切 为 x 的 值 
返回 天 。 它 类 似 或 x 的 计算 ， 除非 两 个 参 
ATANGO_ATAN2G ee 它 类 似 于 y 或 x 的 反正 切 计 算 ， 除 非 两 个 参数 的 符号 均 用 
COT(x) 返回 x 的 余 切 














下 面 对 其 中 的 常用 函数 进行 讲解 ， 并 且 配 合 以 示例 做 详细 说 明 。 


10.2.1 ABS(x) 函 数 


ABS(x) 函 数 用 于 求 绝 对 值 。 


例 10.1 


使 用 ABS(x) 函 数 来 求 5 和 -5 的 绝对 值 ， 其 语句 如 下 。 


select ABS(5),ABS(-5); 
查询 结果 如 图 10.1 所 示 。 


(实例 位 置 ， 光盘 \TMNsIN10\10.1) 
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10.2.2 “FLOOR(x) 函 数 


FLOOR(x) 函 数 返 回 小 于 或 等 于 x 的 最 大 整数 。 

例 10.2 应 用 FLOOR(x) 函 数 求 小 于 或 等 于 1.5 及 -2 的 最 大 整数 ， 其 语句 如 下 。 《实例 位 置 : 光 
盘 \TMNsIN10\10.2) 

select FLOOR(1.5),FLOOR(-2); 


其 查询 结果 如 图 10.2 所 示 。 


5>.PLOORC-225; 





图 10.2 使 用 FLOOR(x) 函 数 求 小 于 或 等 于 数据 的 最 大 整数 
10.2.3 ”RAND() 函 数 


RAND0O 函 数 是 返回 0~1 的 随机 数 。 
例 10.3 运用 RANDO 函 数 ， 获 取 两 个 随机 数 ， 其 语句 如 下 。《〔 实 例 位 置 ， 光盘 \TMNsIN10\10.3) 
Select RAND(),RAND(); 


其 查询 结果 如 图 10.3 所 示 。 





图 10.3 ”使 用 RANDO 函 数 获取 随机 数 
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例 10.4 生成 3 个 1~100 之 间 的 随机 整数 ， 效 果 如 图 10.4 所 示 。 《实例 位 置 : 光盘 \TMNsIM4\10.4) 


row in set (08.80 sec) 





图 10.4 生成 3 个 1~100 之 间 的 随机 整数 


使 用 ROUND(x) 生 成 一 个 与 数 x 最 接近 的 整数 ， 当然, 也 可 以 使 用 FLOOR(x) 来 生成 一 个 小 于 或 者 
等 于 x 的 最 大 整数 。RANDO 产 生 的 是 随机 数 ， 所 以 每 次 执行 的 结果 都 会 是 不 一 样 的 ， 具 体 代码 如 下 。 
Select ROUND(RAND()*100),FLOOR(RAND()*100),CEILING(RAND()*100); 


10.2.4 _PI() 函 数 


PIO 函 数 用 于 返回 圆周 率 。 

例 10.5 使 用 PIO 函 数 获取 圆周 率 ， 其 语句 如 下 。 (实例 位 置 : 光盘 \TMNsIN10\10.5) 
select PI(); 

查询 结果 如 图 10.5 所 示 。 


> select PIC); 


1 3.141593 ! 


row in set (8-BB sec》 





图 10.5 使 用 PIO 函 数 获取 圆周 率 


10.2.5 TRUNCATE(x,y) 函 数 





TRUNCATE(x,y) 函 数 返回 x 保留 到 小 数 点 后 y 位 的 值 。 
例 10.6 使 用 TRUNCATE(x,y) 函 数 返回 2.123 456 7 小 数 点 后 3 位 的 值 ， 其 语句 如 下 。 (实例 位 
置 : 光盘 \TMsIN10\10.6) 


select TRUNCATE(2.1234567,3); 
其 查询 结果 如 图 10.6 所 示 。 
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ysql> select TRUNCATE 





图 10.6 使 用 TRUNCATE(x.y) 函 数 获取 数据 


10.2.6 ROUND(x) 函 数 和 ROUND(x,y) 函 数 


ROUND(x) 函 数 返回 离 x 最 近 的 整数 ， 也 就 是 对 x 进行 四 舍 五 入 处 理 ;，ROUND(x,y) 函 数 返 回 x 保 
留 到 小 数 点 后 y 位 的 值 ， 截 断 时 需要 进行 四 舍 五 入 处 理 。 

例 10.7 使 用 ROUND() 函 数 获取 1.6 和 1.2 最 近 的 整数 ， 使 用 ROUND(x,y) 函 数 获取 1.123 456 
小 数 点 后 3 位 的 值 ， 其 语句 如 下 。《〔 实 例 位 置 ， 光盘 \TMNsN10\10.7) 

select ROUND(1.6),ROUND(1.2),ROUND(1.123456,3); 





查询 结果 如 图 10.7 所 示 。 


ysql> select ROUNDC1.6),.ROUNDC1.2>,ROUNDC1 .123456,.3); 





图 10.7 使 用 ROUND(x) 函 数 和 ROUND(x.y) 函 数 获取 数据 
10.2.7 ”SQRT(x) 函 数 
SQRT(x) 函 数 用 于 求 平方 根 。 
例 10.8 使 用 SQRT(x) 函 数 求 16 和 25 的 平方 根 ， 其 语句 如 下 。( 实 例 位 置 : 光盘 \TMsIN10\10.8) 


select SQRT(16),SQRT(25); 
其 查询 结果 如 图 10.8 所 示 。 


QRT C25>; 





10.8 使 用 SQRT(x) 函 数 求 16 和 25 的 平方 根 
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字符 串 函 数 是 MySQL 中 最 常用 的 一 类 函数 ， 主 要 


函数 


第 10 章 常用 


二 数 


10.3 字符 串 函 数 

















于 处 理 表 中 


表 10.3 MySQL 的 字符 串 函 数 


作 


的 字符 串 ， 其 作用 如 表 10.3 所 示 。 


用 





CHAR LENGTH(s) 


返回 字符 串 s 的 字符 数 





LENGTH(s) 


CONCAT!(s1.s2, …) 


CONCAT WSG.s1.S52. … 


JINSERT(S1.x.len.s2) 
UPPER(S).UCASE(S) 
LOWER(S).LCASE(S) 
LEFT(s.n) 


返回 值 为 字符 串 s 的 长 度 , 单位 为 字 节 。 一 个 多 字 节 字符 算 作 多 字 节 。 这 意 


味 着 对 于 一 个 包含 5 个 2 字 节 字符 的 字符 串 ， 


LENGTHO 的 返回 值 为 


10， 而 CHAR_LENGTHO 的 返回 值 则 为 5 

返回 结果 为 连接 参数 产生 的 字符 串 。 如 有 任何 一 个 参数 为 NULL， 则 返回 值 
为 NULL。 或 诈 有 一 个 或 多 个 参数 。 如 果 所 有 参数 均 为 非 二 进 制 字符 串 ， 则 
结果 为 非 二 进 制 字符 串 。 如 果 自 变量 中 含有 任 一 二 进 制 字符 串 ， 则 结果 为 一 
个 二 进 制 字符 串 。 一 个 数字 参数 被 转化 为 与 之 相等 的 二 进 制 字符 串 格式 ， 若 
要 避免 这 种 情况 , 可 使 用 显 式 类 型 cast, 如 SELECT CONCAT(CAST(int_col 


AS CHAR). char col) 


同 CONCAT(s1,s2, …) 函 数 ， 但 是 每 个 字符 串 直 接 要 加 上 x 
将 字符 串 s2 替换 sl 的 x 位 置 开 始 长 度 为 len 的 字符 串 

将 字符 串 s 的 所 有 字母 都 变 成 大 写字 母 

将 字符 串 s 的 所 有 字母 都 变 成 小 写字 母 

返回 从 字符 串 s 开始 的 n 最 左 字符 


























RIGHT(s.n) 从 字符 串 s 开始 ， 返 回 最 右 n 个 字符 

LPAD(sLlen.s?) 返回 字符 串 Ws 其 左边 由 字符 串 s2 填补 到 len 字符 长 度 。 假 如 sl 的 长 度 
大 于 len ， 则 返回 值 被 缩短 至 len 字符 

RpADGllens2) 返回 字符 串 s1， 其 右边 被 字符 让 5S2 填补 至 len 字符 长 度假 如 字符 申 sl 的 
长 度 大 于 len， 则 返回 值 被 缩短 到 与 len 字符 相同 长 度 

LTRIM(s) 返回 字符 串 s， 其 引导 空格 字符 被 删除 

RTRIM(s) 返回 字符 串 s， 其 结尾 空格 字符 被 删 去 

TRIM(s) 去 掉 字符 串 s 开始 处 和 结尾 处 的 空格 

TRIM(s1 FROM s) 去 掉 字 符 串 s 中 开始 处 和 结尾 处 的 字符 串 sl 

REPEAT(s.n) 将 字符 串 s 重复 0 次 

SPACE(n) 返回 n 个 空格 

REPLACE(s.s1.s2) 用 字符 串 s2 蔡 代 字符 串 s 中 的 字符 串 sl 

STRCMP(s1.s2) 比较 字符 串 sl 和 S2 
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SUBSTRING(snlen) 


函数 


MySQL 从 入 门 到 精通 


作 用 
获取 从 字符 串 s 中 的 第 n 个 位 置 开始 长 度 为 len 的 字符 串 





MID(s.n,len) 


同 SUBSTRING(s.n.len) 





LOCATE(s1.s).POSITION(s1 IN S) 


从 字符 串 s 中 获取 sl 的 开始 位 置 





INSTR(s.s1) 
REVERSE(s) 


查找 字符 串 s1 在 s 中 的 位 置 ， 返 回首 次 出 现 位 置 的 索引 值 
将 字符 串 s 的 顺序 反 过 来 





ELT(n,s1.,s2,…*) 


返回 第 n 个 字符 串 





EXPORT SETCbits.on.off[.separator 
[number of bits]]) 


返回 一 个 字符 串 ， 生 成 规则 如 下 : 针对 bits 的 二 进 制 格式 ， 如 果 其 位 为 1， 
则 返回 一 个 on 值 ; 如 果 其 位 为 0, 则 返回 一 个 o 企 值 。 每 个 字符 串 使 用 separator 
进行 分 隔 ， 默 认 值 为 “.”。number of bits 参数 指定 bits 可 用 的 位 数 ， 默 认 
值 为 64 位 。 例 如 ， 生 成 数字 182 的 二 进 制 (10110110) 替换 格式 ， 以 “@” 
作为 分 隔 符 ， 设 置 有 效 位 为 6 位 。 其 语句 如 下 : select 

EXPORT SET(182,Y",N'.@'.6):; 

其 运行 结果 为 : N@Y@Y@N@Y@Y 





FIELD(s,s1,s2, *…) 
FIND IN SET(s1,s2) 
MAKE SET(x,s1.s2, …) 


10.3.1 


下 面 对 其 中 的 常用 函数 进行 讲解 ， 


INSERT(s1,x,len,s2) 函 数 ) 





例 10.9 使 用 INSERT 函数 将 


\TMNsM\10\10.9) 
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select INSERT('mrkej',3,2,'book'); 





字符 串 sl 中 x 位 置 开始 长 度 为 len 的 字 


返回 第 一 个 与 字符 串 s 匹配 的 字符 串 的 位 置 
返回 在 字符 串 s2 中 与 s1 匹配 的 字符 串 的 位 置 
按 x 的 二 进 制 数 从 s1.s2. ….sn 中 选取 字符 串 





并 且 结 合 例子 做 详细 说 明 。 


INSERT(s1,x,len,s2) 函 数 


符 串 用 字符 串 s2 替换 。 
mrkj 字符 串 中 的 二 替换 为 book， 其 语句 如 下 。 (实例 位 置 ， 光盘 





替换 后 的 查询 结果 如 图 10.9 所 示 。 


1 mrbook 


1 row in 


图 10.9 


set 《B-88 sec) 





使 用 INSERT(s1.x.len.s2) 函 数 蔡 换 指定 字符 串 


第 10 章 常用 函数 


10.3.2 UPPER(s) 函 数 和 UCASE(s) 函 数 


UPPER(s) 函 数 和 UCASE(s) 函 数 将 字符 串 s 的 所 有 字母 变 成 大 写字 母 。 

例 10.10 下 面 使 用 UPPER(s) 函 数 和 UCASE(s) 函 数 将 mrbccd 字 符 串 中 的 所 有 字母 变 成 大 写字 母 ， 
其 语句 如 下 。 《实例 位 置 : 光盘 \TMNsIN10\10.10) 

select UPPER(mrbccd'),UCASE(mrbccd'); 


其 转换 后 的 结果 如 图 10.10 所 示 。 





mysql》 select UPPERC’mrbccd’ >-UCRSECmrbccd' ?3 


PPERC’ mrhbcc Cnrbcc: 


! MRBCCD 


row in set 0.88 sec》 





vsql> 


图 10.10 使 用 UPPER(s) 函 数 和 UCASE(s) 函 数 将 mrbccd 字符 串 中 的 所 有 字母 变 成 大 写字 母 
10.3.3 LEFT(s,n) 函 数 


LEFT(s.n) 函 数 返 回 字符 串 s 的 前 n 个 字符 。 
例 10.11 应 用 LEFT 函数 返回 mrbccd 字符 串 的 前 两 个 字符 ， 其 语句 如 下 。 (实例 位 置 ， 光盘 
VTMNsINIO\10.11) 


select LEFT(mrbccd',2); 
其 截取 结果 如 图 10.11 所 示 。 





图 10.11 使 用 LEFT(s.n) 函 数 返回 指定 字符 


10.3.4 “RTRIM(s) 函 数 


RIRIM(s) 函 数 将 去 掉 字 符 串 s 结尾 处 的 空格 。 


pl 


MySQL 从 入 门 到 精通 


例 10.12 应 用 RTRIM 函数 去 掉 mr 结尾 处 的 空格 , 其 语句 如 下 .( 实 例 位 置 :光盘 \TMNsN10\10.12) 
select CONCAT(+",RTRIM(' mr "),+"); 
其 结果 如 图 10.12 所 示 。 





图 10.12 使 用 RTRIM(S) 函 数 去 掉 mr 结尾 处 的 空格 
10.3.5 SUBSTRING(s,n,len) 函 数 


SUBSTRING(snlen) 函 数 从 字符 串 s 的 第 n 个 位 置 开 始 获取 长 度 为 len 的 字符 串 。 
例 10.13 下 面 使 用 SUBSTRING 函数 从 mrbccd 字符 串 的 第 三 位 开始 获取 4 个 字符 ,结果 如 图 10.13 
所 示 。 【实例 位 置 : 光盘 \TMNsM0\10.13) 





图 10.13 ”使 用 SUBSTRING(s.n.len) 函 数 获取 指定 长 度 字符 串 
10.3.6 ”REVERSE(s) 函 数 


REVERSE(s) 函 数 将 字符 串 s 的 顺序 反 过 来 。 
例 10.14 下 面 使 用 REVERSE 函数 将 mrbccd 字符 串 的 顺序 反 过 来 ， 结 果 如 图 10.14 所 示 。 〈 实 
例 位 置 ， 光盘 \TMNsN10\10.14) 





ysql> select REVERSEC’mrbccd’); 


1 REVERSEC’ mrh 














图 10.14 ”使 用 REVERSE(s) 函 数 将 mrbccd 字符 串 的 顺序 反 过 来 
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10.3.7 ”FIELD(s,s1,s2,…) 函 数 








FIELD(s,s1,s2,…) 函 数 返回 第 一 个 与 字符 串 s 匹配 的 字符 串 的 位 置 。 
例 10.15 应 用 FIELD 函数 返回 第 一 个 与 字符 串 mr 匹配 的 字符 串 的 位 置 ， 结 果 如 图 10.15 所 示 。 
(实例 位 置 ， 光盘 \TMNsN\M10\10.15) 


























oy in set (0.00 sec) 





图 10.15 合用 FIELD(s.s1.s2.…) 函 数 返回 第 一 个 与 字符 串 mr 匹配 的 字符 串 位 置 
10.3.8 LOCATE(s1,s) 函 数 、POSITION(s1 IN s) 函 数 和 INSTR(s,s1) 函 数 


在 MySQL 中 ,可 以 通过 LOCATE(s1,s)、POSITION(sl IN s) 和 INSTR(s,s1) 函 数 获取 子 字符 串 相 匹 
配 的 开始 位 置 。 这 3 个 函数 的 语法 格式 如 下 。 

(1) LOCATE(s1,s): 表示 子 字符 串 sl 和 在 字符 串 s 中 的 开始 位 置 

(2) POSITION(sl IN s): 表示 子 字符 串 sl 在 字符 串 s 中 的 开始 位 置 。 

(3) INSTR(s,s1): 表示 子 字符 串 sl 在 字符 串 s 中 的 开始 位 置 


全 o 注 总 
在 使 用 这 3 个 函数 时 ， 前 两 个 函数 LOCATE(s1.s) 和 POSITION(s1 IN s) 的 参数 中 ， 是 把 子 字符 
串 作 为 第 一 个 参数 ， 后 第 三 个 函数 INSTR(s,s1) 则 需要 把 子 字符 串 作 为 第 二 个 参数 ， 这 一 点 一 定 不 
要 记 错 。 


例 10.16 返回 字符 串 “me "在 字符 串 ' You love me .He love me ' 中 第 一 次 出 现 的 位 置 .效果 如 图 10.16 
所 示 。【〔 实 例 位 置 光盘 \TMNsM0\10.16) 











图 10.16 字符 串 函 数 的 使 用 
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对 于 字符 串 函 数 中 的 LOCATE(s1,s),POSITION(sl1 IN s) 是 从 字符 串 s 中 获取 sl 的 开始 位 置 ， 具 体 
代码 如 下 。 


Select LOCATE(me', 'You love me.He love me. "); 
Select POSITION(Cme' IN You love me.He love me."); 


10.4 日期 和 时 间 函 数 


日 期 和 时 间 函 数 是 MySQL 中 另 一 最 常用 的 函数 ， 主 要 用 于 对 表 中 的 日 期 和 时 间 数 据 的 处 理 。 
MySQL 内 置 的 日 期 时 间 函 数 及 作用 如 表 10.4 所 示 。 
表 10.4 MySQL 的 日 期 和 时 间 函 数 
































函数 作 用 
CURDATE0.CURRENT DATEO 返回 当前 日 期 
CURTIMEO.CURRENT TIMEO 返回 当前 时 间 
DA ATE TCA | 电光 
UNIX_TIMESTAMPO 以 UNIX 时 间 稚 的 形式 返回 当前 时 间 
UNIX_TIMESTAMP(d) 将 时 间 d 以 UNIX 时 间 戳 的 形式 返回 
FROM_UNIXTIME(d) 把 UNIX 时 间 戳 的 时 间 转 换 为 普通 格式 的 时 间 
UTC DATE0 返回 UTC (Universal Coordinated Time， 国 际 协 调 时 间 ) 日 期 
UTC TIMEO 返回 UTC 时 间 
MONTH(d) 返回 日 期 d 中 的 月 份 值 ， 范 围 是 1~12 
MONTHNAME(d) 返回 日 期 d 中 的 月 份 名 称 ， 如 January、February 等 
DAYNAME(d) 返回 日 期 4 是 星期 几 ， 如 Monday、Tuesday 等 
DAYOFWEEK(d) 返回 日 期 4 是 星期 几 ，1 表示 星期 日 ，2 表示 星期 一 等 
WEEKDAY(d) 返回 日 期 4 是 星期 几 ，0 表示 星期 一 ，1 表示 星期 二 等 
WEEK(d) 计算 日 期 4 是 本 年 的 第 几 个 星期 ， 范 围 是 0~53 
WEEKOFYEAR(d) 计算 日 期 4 是 本 年 的 第 几 个 星期 ， 范 围 是 1~53 
DAYOFYEAR(d) 计算 日 期 d 是 本 年 的 第 几 天 
DAYOFMONTH(d) 计算 日 期 d 是 本 月 的 第 几 天 
YEAR(d) 返回 日 期 4 中 的 年 份 值 
QUARTER(d) 返回 日 期 4 是 第 几 季度 ， 范 围 是 1~4 
HOUR(D 返回 时 间 t 中 的 小 时 值 
MINUTE(O) 返回 时 间 t 中 的 分 钟 值 
SECOND(D 返回 时 间 t 中 的 秒 钟 值 
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续 表 
函数 作 用 
EXTRACT(type FROM d) 从 日 期 4 中 获取 指定 的 值 ， type 指定 返回 的 值 ， 如 YEAR、HOUR 
等 将 时 间 转换 为 秒 
TIME TO SEC(t) 将 时 间 t 转 换 为 秒 
SEC TO TIME(s) 将 以 秒 为 单位 的 时 间 s 转换 为 时 分 秒 的 格式 
TO DAYS(d) 计算 日 期 4~0000 年 1 月 1 日 的 天 数 
FROM DAYS(n) 计算 从 0000 年 1 月 1 日 开始 n 天 后 的 日 期 
DATEDIFF(d1.d2) 计算 日 期 d1~d2 之 间 相 隔 的 天 数 
ADDDATE(dn) 计算 起 始 日 期 4 加 上 nm 天 的 日 期 
ADDDATE(d.INTERVAL expr type) 计算 起 始 日 期 4 加 上 一 个 时 间 段 后 的 日 期 
DATE ADD(dINTERVAL expr type) 同 ADDDATE(d.INTERVAL n type) 
SUBDATE(dn) 计算 起 始 日 期 d 减 去 n 天 后 的 日 期 
SUBDATE(d.INTERVAL expr type) 计算 起 始 日 期 d 减 去 一 个 时 间 段 后 的 日 期 
ADDTIME(t.n) 计算 起 始 时 间 t 加 上 nm 秒 的 时 间 
SUBTIME(t.n) 计算 起 始 时 间 t 减 去 n 秒 的 时 间 
DATE FROMAT(dD 按照 表达 式 f 的 要 求 显示 日 期 d 
TIME FROMAT(tf) 按照 表达 式 f 的 要 求 显示 时 间 f 
GET FORMAT(type.s) 根据 字符 串 s 获取 type 类 型 数据 的 显示 格式 


10.4.1 CURDATE() 函 数 和 CURRENT_DATE() 函 数 


CURDATE0O 函 数 和 CURRENT_DATE0 函 数 用 于 获取 当前 日 期 。 

例 10.17 下 面 使 用 CURDATEO 函 数 和 CURRENT _DATEO 函 数 获取 当前 日 期 , 其 语句 如 下 。( 实 
例 位 置 ， 光 盘 \TMNsIN10\10.17) 

select CURDATE(),CURRENT_DATE(); 


其 查询 结果 如 图 10.17 所 示 。 














10.17 使 用 CURDATE0 和 CURRENT DATE0 函 数 获取 当前 日 期 





17S 


MySQL 从 入 门 到 精通 


10.4.2 ”CURTIME() 函 数 和 CURRENT_TIME() 函 数 


CURTIME0O 函 数 和 CURRENT_TIME0O 函 数 用 于 获取 当前 时 间 。 
例 10.18 下面 使 用 CURTIMEO 函 数 和 CURRENT_TIMEO 函 数 获取 当前 时 间 ， 其 语句 如 下 。( 实 


例 位 置 : 光盘 \TMsN10\10.18) 
select CURTIME(),CURRENT_TIME(); 


其 查询 结果 如 图 10.18 所 示 。 


ect CURTIME()> .CURRENT_TIMEC7; 


1 CURTIMEC> SURRENT_TIMEC> } 


:99:16 


I row in set 《8.B1 sec》 


nysql> ~ 





图 10.18 使 用 CURTIMEO 函 数 和 CURRENT_TIMEO 函 数 获取 当前 时 间 


10.4.3 NOW() 函 数 


NOW0 函 数 获取 当前 日 期 和 时 间 。 还 有 URRENT_TIMESTAMPO 函 数 、LOCALTIMEO 函 数 、 
SYSDATE0O 函 数 和 LOCALTIMESTAMPO 函 数 也 同样 可 以 获取 当前 日 期 和 时 间 。 

例 10.19 下 面 使 用 NOWO 函 数 .CURRENT_TIMESTAMPO 函 数 .LOCALTIME0O 函 数 .SYSDATEQO 
函数 和 LOCALTIMESTAMPO 函 数 来 获取 当前 日 期 和 时 间 ， 其 语句 如 下 。【〔 实 例 位 置 ， 光盘 
\IMN\sI\MO\10.19) 

select NOW(),CURRENT_TIMESTAMP()LOCALTIME(),SYSDATE(); 

运行 结果 如 图 10.19 所 示 。 


URRENT_TIMESTAMPC> } LOCALTIMEC> ! SYSDRTEC》 
! LOCALTIMESTAMPC> 


row in set (0.80 sec) 


ysql> 


图 10.19 使 用 NOWO、CURRENT_TIMESTAMPO 等 函数 获取 当前 日 期 和 时 间 
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10.4.4 DATEDIFF(d1,d2) 函 数 


DATEDIFF(d1,d2) 用 于 计算 日 期 dl 与 d2 之 间 相 隔 的 天 数 。 

例 10.20 使 用 DATEDIFF(dl1,d2) 函 数 计算 2011-07-05 与 2011-07-01 之 间 相 隔 的 天 数 ， 其 语句 如 
下 。《〈 实 例 位 置 ， 光盘 \TMNsI\10\10.20) 

select DATEDIFF('2011-07-05',2011-07-01"); 


结果 如 图 10.20 所 示 。 


2811-87-8@1’ >; 


I row ;in set 《8.860 sec) 





husgl> 


图 10.20 使 用 DATEDIFF(d1,d2) 函 数 计算 2011-07-05 与 2011-07-01 之 间 相隔 的 天 数 
10.4.5 ADDDATE(d,n) 函 数 


ADDDATE(dn) 用 于 返回 起 始 日 期 4 加 上 mn 天 的 日 期 。 
例 10.21 使 用 ADDDATE(dn) 函 数 返 回 2011-07-01 加 上 3 天 的 日 期 ， 结 果 如 图 10.21 所 示 。 ( 实 
例 位 置 ， 光盘 \TMNsN10\10.21) 


Inysql> select ADDDATEC’2@11-87-81’ .3); 
1 ADDDATEC’2811-87-81’ .3> } 
! 2911-97-94 


i row in set 0.80 sec)» 





图 10.21 使 用 ADDDATE(d.n) 函 数 返回 2011-07-01 加 上 3 天 的 日 期 


10.4.6 ADDDATE(d,INTERVAL expr type) 函 数 





ADDDATE(d,INTERVAL exprtype) 函 数 返 回 起 始 日 期 4 加 上 一 个 时 间 段 后 的 日 期 。 

例 10.22 使 用 ADDDATE(d,INTERVAL expr type) 函 数 返 回 2011-07-01 加 上 一 年 两 个 月 后 的 日 
期 ， 其 语句 如 下 。《〔 实 例 位 置 ， 光盘 \TMNsI\10\10.22) 

select ADDDATE('2011-07-01',INTERVAL,'12' YEAR_MONTH); 
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其 运行 结果 如 图 10.22 所 示 。 


YERR_MHONTH2; 





图 10.22 使 用 ADDDATE(d.INTERVAL expr type) 函 数 返回 2011-07-01 加 上 一 年 两 个 月 后 的 日 期 
10.4.7 SUBDATE(d,n) 函 数 


SUBDATE(dn) 函 数 返 回 起 始 日 期 d 减 去 n 天 的 日 期 。 
例 10.23 使 用 SUBDATE(dm) 函 数 返 回 2011-07-01 减 去 6 天 后 的 日 期 , 结果 如 图 10.23 所 示 。( 实 
例 位 置 ， 光盘 \TMNsN10\10.23) 


1 2811-86-25 


1 row in set (0.80 sec) 
nysgl> 


图 10.23 ”使 用 SUBDATE(d.n) 函 数 返 回 2011-07-01 减 去 6 天 后 的 日 期 





10.5 条 件 判 断 函数 


条 件 函 数 用 来 在 SQL 语句 中 进行 条 件 判断 。 根 据 不 同 的 条 件 ， 执 行 不 同 的 SQL 语句 。MySQL 支 
持 的 条 件 判 断 函数 及 作用 如 表 10.5 所 示 。 
表 10.5 _ MySQL 的 条 件 判断 函数 















































函数 作 用 

IF(expr.v1.v2) 如 果 表 达 式 expr 成 立 ， 则 执行 V1; 否则 执行 V2 

IFNULL(v1.v2) 如 果 v1 不 为 空 ， 则 显示 v1 的 值 ， 否 则 显示 v2 的 值 
CASE 表示 函数 开始 , END 表示 函数 结束 。 如 果 表 达 式 exprl 成 立 ， 

CASE WHEN exprl THEN v1 [WHEN expr2 则 返回 v1 的 值 ， 如 果 表 达 式 expr2 成 立 ， 则 返回 v2 的 值 。 以 此 类 

THEN v2 …][ELSE vn] END 推 ， 最 后 遇 到 else 时 ， 返 回 vn 的 值 。 它 的 功能 与 PHP 中 的 switch 
语句 类 似 

ER CASE 表示 函数 开始 ， END 表示 函数 结束 。 如 果 表 达 式 expr 取 值 

THEN v2 …][ELSE wm END 为 el， 则 返回 v1 的 值 ， 因果 表达 式 expe 取 值 为 e2， 则 返回 v2 的 
值 ， 以 此 类 推 ， 最 后 遇 到 ELSE， 则 返回 vn 的 值 


178 


第 10 章 常用 函数 


例 10.24 查询 编程 词典 业绩 信息 表 ， 如 果 业 绩 超过 100 万 ， 则 输出 “Very Good”; 如 果 业 绩 小 
于 100 万 大 于 10 万 ， 则 输出 “Popularly”; 否则 输出 “Not Good”。 其 语句 如 下 。〔 实 例 位 置 ， 光盘 
VTMNsINI0\10.24) 


select id,grade, CASE WHEN grade>1000000 THEN Very Good WHEN grade<1000000 and grade >=100000 
THEN 'Popularly ELSE 'Not Good' END level from tb_bccd; 


其 查询 结果 如 图 10.24 所 示 。 


WHEN grade<1BB 
1 from th_hccd; 





图 10.24 条 件 判断 函数 的 应 用 
10.6 系统 信息 函数 


系统 信息 函数 用 来 查询 MySQL 数据 库 的 系统 信息 。 例 如 ， 查 询 数 据 库 的 版 本 ， 查 询 数 据 库 的 当 
前 用 户 等 。 如 表 10.6 所 示 为 各 种 系统 信息 函数 的 作用 。 
表 10.6 ”MySQL 的 系统 信息 函数 




















函数 作 用 示 例 
VERSIONO 获取 数据 库 的 版 本 号 select VERSIONO: 
CONNECTION IDO 获取 服务 器 的 连接 数 select CONNECTION _IDO: 
DATABASEO.SCHEMAO 获取 当前 数据 库 名 select DATABASEO.SCHEMANO: 
USERO.SYSTEM_USERO.SESSION_USERO ”| 获取 当前 用 户 select USERO.SYSTEM_USERO; 
CURRENT USERO.CURRENT USER 获取 当前 用 户 select CURRENT USERO: 
CHARSET(str) 获取 字符 串 str 的 字符 集 select CHARSET(mrsoft'): 
COLLATION(str) 获取 字符 串 str 的 字符 排列 方式 “| select COLLATION(Cmrsoft): 
LAST _ INSERT ID0 RS selectLAST _ INSERT ID0: 

AUTO INCREMENT 值 








10.6.1 获取 MySQL 版 本 号 、 连 接 数 和 数据 库 名 的 函数 











VERSIONO 函 数 返回 数据 库 的 版 本 号 ; CONNECTION IDO 函 数 返 回 服务 器 的 连接 数 ,也 就 是 到 现 
在 为 止 MySQL 服务 的 连接 次 数 ; DATABASE0 函 数 和 SCHEMAO 函 数 返回 当前 数据 库 名 。 
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例 10.25 下面 将 演示 VERSIONO、CONNECTION IDO、DATABASEO 和 SCHEMAO 4 个 函数 
的 用 法 ， 如 图 10.25 所 示 。 《实例 位 置 : 光盘 \TMNsIM0\10.25) 


row in set 《B.BB sec? 





图 10.25 获取 MySQL 版 本 号 、 连 接 数 和 数据 库 名 得 函数 


其 中 ，VERSIONO 函 数 返 回 的 版 本 号 为 “5.6.20”; CONNECTOIN_IDO 函 数 返回 的 连接 数 为 1; 
DATABASE0 函 数 和 SCHEMAO 函 数 返回 的 当前 数据 库 名 是 test。 


10.6.2 ”获取 用 户 名 的 函数 


USERO、SYSTEM_USERO、SESSION_USERO、CURRENT_USERO 和 CURRENT_USER 这 几 个 
函数 可 以 返回 当前 用 户 的 名 称 。 
例 10.26 查询 当前 用 户 的 用 户 名 ， 效 果 如 图 10.26 所 示 。 (实例 位 置 ， 光盘 \TMNsIN10\10.26) 


EM_USERC 


CURRENT | 


1 rootBlocalho rootBlocalho: 


row in set 0.88 sec》 





ysql> 。 
图 10.26 获取 用 户 名 的 函数 


结果 显示 ， 当 前 用 户 的 用 户 名 为 root。localhost 是 主机 名 。 因 为 服务 器 和 客户 端 在 一 台 机 器 上 ， 所 
以 服务 器 的 主机 名 为 localhost。 用 户 名 和 主机 名 之 间 用 符号 “@” 进 行 连接 。 
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10.6.3 ”获取 字符 串 的 字符 集 和 排序 方式 的 函数 


CHARSET(str) 函 数 返 回 字符 串 str 的 字符 集 ， 














COLLATION(str) 函 数 返 回 字符 串 str 的 字符 排列 方式 。 


例 10.27 查看 字符 串 'aa' 的 字符 集 和 字符 串 排 序 方式 ， 效 果 如 图 10.27 所 示 。 


TMN\sN\10\10.27) 


加 密 函 数 是 MySQL 中 用 来 对 数据 进行 加 密 的 函数 。 因 
他 人 看 到 ， 所 以 就 可 以 通过 加 密 的 方式 来 使 这 些 数 据 变 成 看 似 乱码 的 数据 。 例 如 ， 用 户 


Inysql> SELECT CHARSETC’aa’ 
1 CHARSETC’aa’> 1 COLLATIONC’aa’> ! 


1 utf8_general ci ! 


10.7 


行 加 密 。 各 种 加 密 函 数 的 作用 如 表 10.7 所 示 。 
表 10.7 MySQL 的 加 密 函 数 


- 般 情 况 下 这 个 字符 集 就 是 系统 的 默认 字符 集 ; 


《实例 位 置 ， 光盘 \ 


) ,COLLATIONC’ aa’ >; 





图 10.27 获取 字符 串 的 字符 集 和 排序 方式 的 函数 


数 


加 密 未 考 


为 数据 库 中 有 些 很 敏感 的 信息 不 希望 被 其 


密码 就 应 该 进 








函数 作 用 示例 
对 字符 串 str 进行 加 密 。 经 此 函数 加 密 | 
PASSWORDGstD 后 的 数据 是 不 可 道 的。 其 经 常用 于 对 用 | 对 字符 中 mrsof 进行 加 密 ， 其 语 匀 如 下 
户 注册 的 密码 进行 加 密 处 理 select PASSWORD('mrsoft'’): 
a | 使 用 MD50 函 数 对 mrsoft 字符 串 进行 加 密 ， 其 
生字 条 个 。 经 常 0 
MDS(stn) 对 字符 串 str 进行 加 密 。 经 常用 于 对 普 语句 如 下 ， 


通 数据 进行 加 密 


select MDSCmrsoft): 





ENCODE(str.pswd_str) 


使 用 字符 串 pswd_str 来 加 密 字 符 串 str。 
加 密 的 结果 是 一 个 二 进 制 数 ， 必 须 使 用 


使 用 字符 串 mr 对 mrsoft 进行 加 密 处 理 ， 其 语 
句 如 下 : 


BLOB 类 型 的 字段 来 保存 它 select ENCODE(mrsoft mr): 


DECODE(crypt_str.pswd_ 
str) 





使 用 字符 串 pswd_str 来 为 crypt_str 解密 。 
crypt_str 是 通过 ENCODE(str.pswd_str) 
加 密 后 的 二 进 制 数据 。 字 符 串 pswd_str 
应 该 与 加 密 时 的 字符 串 pswd_str 是 相 
同 的 





应 用 DECODE 函数 对 经 过 ENCODE 函数 加 密 
的 字符 串 进行 解密 ， 其 语句 如 下 : 
select DECODE(ENCODE(mrso 人 .mr).mr): 
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10.7.1 加 密 函 数 PASSWORD(str) 





PASSWORD(str) 函 数 可 以 对 字符 串 str 进行 加 密 。 一 般 情况 下 , PASSWORD(str) 函 数 主要 是 用 来 给 
用 户 的 密码 加 密 的 。 

例 10.28 使 用 PASSWORD(stD 函 数 为 字符 串 'abcd 加 密 ， 效 果 如 图 10.28 所 示 。 实例 位 置 : 光 
盘 \TMNsIN10\10.28) 





图 10.28 加 密 函 数 PASSWORD(tD 





结果 显示 , 字符 串 “abcd” 加 密 后 的 结果 是 “*A154C2565E9E7F94BFC08A1FE702624ED8EFDA”。 
PASSWORD(stD 函 数 加 密 是 不 可 逆 的 。 


[234 

PASSWORD(stD 函 数 经 常用 来 给 密码 加 密 。MYSQL 用 户 需要 设置 密码 ， 用 户 不 能 将 未 加 密 的 
密码 直接 存储 到 MySQL 的 user 表 中 。 因 为 登录 MySQL 数据 库 时 ， 数 据 库 系统 会 将 输入 的 密码 先 
通过 PASSWORD(stD 函 数 加 密 ， 然 后 与 数据 库 中 的 密码 进行 比较 ， 匹 配 成 功 后 才 可 以 登录 。 


10.7.2 ”加 密 函 数 MD5(str) 


MD5(str) 函 数 可 以 对 字符 串 str 进行 加 密 。MD5(str) 函 数 主要 对 普通 的 数据 进行 加 密 。 
例 10.29 使 用 MD5(str) 函 数 为 字符 串 'abcd' 加 密 ， 效 果 如 图 10.29 所 示 。【〔 实 例 位 置 ， 光 盘 \ 
TM\sI\M10\10.29) 








图 10.29 加密 函数 MD5(str) 





结果 显示 ， 字 符 串 abcd 的 MD5 值 为 e2fc714c4727ee9395f324cd2e7f331f。 
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10.8 其 他 函数 


MySQL 中 除了 上 述 内 置 函数 以 外 ， 还 包含 很 多 函数 。 例 如 ， 数 字 格 式 化 函数 FORMAT(x,n)， 人 P 
地 址 与 数字 的 转换 函数 INET_ ATON(ip)， 还 有 加 锁 函 数 GET LOCTCame:time) 、 解 锁 函 数 
RELEASE_LOCK(name) 等 。 在 表 10.8 中 罗列 了 MySQL 中 支持 的 其 他 函数 。 


函数 
FORMAT(x.n) 
ASCII(s) 
BIN(x) 
HEX(x) 
OCT(x) 
CONV(x.f1,f2) 
INET ATON(P) 
INET_NTOA(N) 


GET LOCT(name.time) 


RELEASE LOCK(name) 


IS FREE LOCK(name) 


BENCHMARK(count.expr) 


CONVERT(s USING cs) 


表 10.8 MySQL 的 其 他 函数 


作 用 
将 数字 x 进行 格式 化 ， 将 x 保留 到 小 数 点 后 n 位 。 这 个 过 程 需要 进行 四 舍 五 入 
ASCII(s) 返 回 字符 串 s 的 第 一 个 字符 的 ASCI 码 
BIN(x) 返 回 x 的 二 进 制 编码 
HEX(x) 返 回 x 的 十 六 进 制 编码 
OCT(%) 返 回 x 的 八进制 编码 
CONV(x.f1, 人 2) 将 x 从 和 进 制 数 变 成 亿 进 制 数 
INET_ATON(IP) 函 数 可 以 将 IP 地 址 转换 为 数字 表示 
INET_NTOA(N) 函 数 可 以 将 数字 n 转换 成 他 的 形式 
GET_LOCT(name,time) 函 数 定义 一 个 名 称 为 name、 持 续 时 间 长 度 为 time 秒 的 锁 。 锁 
定 成 功 ， 返 回 1; 如 果 尝试 超时 ， 返 回 0， 如 果 遇 到 错误 ， 返 回 NULL 
RELEASE_ LOCK(name) 函 数 解除 名 称 为 name 的 锁 。 如 果 解 锁 成 功 ， 返回 1; 如 果 尝 
试 超 时 ， 返 回 0， 如 果 解 锁 失 败 ， 返 回 NULL 
IS_FREE_ LOCK(name) 函 数 判断 是 否 使 用 名 为 name 的 锁 。 如 果 使 用 ， 返 回 0， 否则 
返回 1 
将 表达 式 expr 重复 执行 count 次 ， 然 后 返回 执行 时 间 。 该 函数 可 以 用 来 判断 MySQL 
处 理 表达 式 的 速度 
将 字符 串 s 的 字符 集 变 成 cs 




















CAST(x AS type) 


将 x 变 成 type 类 型 , 这 两 个 函数 只 对 BINARY、 CHAR、 DATE、 DATETIME、 TIME、 
SIGNED INTEGER、UNSIGNED INTEGER 这 些 类 型 起 作用 。 但 两 种 方法 只 是 改变 
了 输出 值 的 数据 类 型 ， 并 没有 改变 表 中 字段 的 类 型 


10.8.1 格式 化 函数 FORMAT(x,n) 





FORMAT(x.n) 函 数 可 以 将 数字 x 进行 格式 化 ， 将 x 保留 到 小 数 点 后 n 位 。 这 个 过 程 需要 进行 四 舍 
五 入 。 例 如 ，FORMAT(2.356,2) 返 回 的 结果 将 会 是 2.36; FORMAT(2.353,2) 返 回 的 结果 将 会 是 2.35。 
例 10.30 使 用 FORMAT(x,n) 函 数 来 将 235.345 6 和 235.345 4 进行 格式 化 ， 都 保留 到 小 数 点 后 3 


位 ， 效 果 如 图 10.30 所 示 。 


(实例 位 置 ， 光盘 \TMsN10\10.30) 
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《0.83 sec) 





图 10.30 格式 化 函数 FORMAT 
结果 显示 ，235.345 6 格式 化 后 的 结果 是 235.346; 235.345 4 格式 化 后 的 结果 是 235.345。 这 个 数 都 
保留 到 小 数 点 后 3 位 ， 而 且 都 进行 了 四 舍 五 入 处 理 。 
- 
9 注意 
FORMAT(%,n) 函 数 可 以 将 XxX 保留 到 小 数 点 后 nn 位 。 在 格式 化 过 程 中 需要 进行 四 使 五 入 的 操作 。 
FORMAT(x.D) 函 数 与 ROUND(&.y) 函 数 返回 X 保 留 到 小 数 点 后 了 位 的 值 。 截断 时 需要 进行 四 舍 五 入 
处 理 。 














10.8.2 ”改变 字符 集 的 函数 


CONVERT(s USING cs) 函 数 将 字符 串 s 的 字符 集 变 成 cs。 
例 10.31 将 字符 串 'ABC' 的 字符 集 变 成 gbk， 效 果 如 图 10.31 所 示 。 【实例 位 置 : 光盘 \ 
TMsINLO\10.31) 





nysql> SELECT CHARSETC’ABC’)>,.CHARSETCCONUERTC’ABC’ USING gbk??; 


RSETC’ABC’> ! CHARSETCCONVERTC’ABC’ USING gbk?》! 
! utf8 


I row in set C0.82 





ySql> 


图 10.31 改变 字符 集 的 函数 
10.8.3 ”改变 字段 数据 类 型 的 函数 


CASTG AS type) 和 CONVERT(%,type) 这 两 个 函数 将 x 变 成 type 类 型 。 这 两 个 函数 只 对 BINARY、 
CHAR、DATETIME、TIME、SIGNED INTEGER、UNSIGNED INTEGER 这 些 类 型 起 作用 。 但 两 种 方 
法 只 是 改变 了 输出 值 的 数据 类 型 ， 并 没有 改变 表 中 字段 的 类 型 。 

例 10.32 ”下面 Cl 表 中 的 times 字段 为 DATETIME 类 型 ， 将 其 变 为 DATE 类 型 或 者 TIME 类 型 ， 
效果 如 图 10.32 所 示 。 《实例 位 置 : 光盘 \TMsI\10\10.32) 
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TIME> FROM C1; 





图 10.32 ”改变 字段 数据 类 型 的 函数 
结果 显示 ，times 字段 原来 的 取 值 是 2014-09-16 11:05:30， 这 是 DATETIME 类 型 ; CAST(times AS 
DATE) 返 回 的 结果 是 2014-09-16， 这 说 明 类 型 已 经 变 成 了 DATE 型 ， CONVERT(times,TIME) 返 回 的 结 


果 是 11:05:30， 这 说 明 类 型 已 经 变 成 了 TIME 类 型 





10.9 小 结 


本 章 介 绍 了 MySQL 数据 库 提供 的 内 部 函数 ， 包 括 数 学 函数 、 字 符 串 函数 、 日 期 和 时 间 函 数 、 条 
件 判断 函数 、 系 统 信息 函 数 和 加 密 函 数 字符 串 函数 、 日 期 和 时 间 函 数 是 本章 的 重 上 容 ， 条件 判 
断 函 数 是 本 章 的 难点 ， 因 为 条 件 判 断 函数 涉及 很 多 条 件 判 断 和 跳 转 的 语句 。 这 些 函 数 通常 与 SELECT 
语句 一 起 使 用 ， 用 来 方便 用 户 的 查询 。 同 时 ，INSERT、UPDATE、DELECT 语句 和 条 件 表达 式 也 可 以 
使 用 这 些 函 数 。 











10.10 ”实践 与 练习 


1. 编写 SQL 语句 ， 实 现 利 用 函数 来 查看 当前 数据 库 的 版 本 号 、 当 前 数据 库 的 名 称 和 当前 的 用 户 。 
pe 光 担 \TMNsIM0\10.33) 
， 编写 SQL 语句 ， 应 用 INSTRO 函 数 返回 字符 串 “me” 在 字符 串 “You love me .He love me” 中 
第 pe 置 。 〈 答 案 位 置 : 光盘 \TMNsIM0\10.34) 
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NA 


索 引 


("视频 讲解 .22 分 钟 ) 


索引 是 一 种 将 殊 的 数据 库 结构 ， 是 提高 数据 库 性 能 的 重要 方式 ,可 以 用 来 快速 
得 询 数据 库 表 中 的 将 定 记录 ，MyYSQL 中 所 有 的 数据 类 型 都 可 以 被 索引 。My5QL 的 
索引 包括 普通 索引 、 唯 一 性 索引 、 全 文 索引 、 单 列 索引 、 多 列 索引 和 空间 索引 等 。 
本 章 将 介绍 索引 的 概念 、 作 用 、 不 同类 别 ， 用 不 同 的 方法 创建 索引 以 及 删除 索引 的 
方法 等 。 

通过 阅读 本 章 ， 读 者 可 以 : 

MM 了 解 MySQL 索引 的 概念 

了解 MySQL 索引 的 分 类 
MI 党 松 在 建立 数据 表 时 创建 索引 的 方法 
MI 党 氢 在 已 建立 的 数据 表 中 创建 索引 的 方法 
MI 党 氢 修 改 数 据 表 结构 添加 案 引 的 方法 
MI 党 要 删除 索引 的 方法 
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11.1 索引 概述 


在 MySQL 中 ， 索 引 由 数据 表 中 一 列 或 多 列 组 合 而 成 ， 创 建 索引 的 目的 是 为 了 优化 数据 库 的 查询 
速度 。 其 中 ， 用 户 创 建 的 索引 指向 数据 库 中 具体 数据 所 在 位 置 。 当 用 户 通过 索引 查询 数据 库 中 的 数据 
时 ， 不 需要 遍历 所 有 数据 库 中 的 所 有 数据 ， 大 幅度 提高 了 查询 效率 。 


11.1.1 _ MySQL 索引 概述 


索引 是 一 种 将 数据 库 中 单列 或 者 多 列 的 值 进行 排序 的 结构 。 应 用 索引 ， 可 以 大 幅度 提高 查询 的 
速度 。 

用 户 通 过 索引 查询 数据 ， 不 但 可 以 提高 查询 速度 ， 也 可 以 降低 服务 器 的 负载 。 用 户 查 询 数据 时 ， 
系统 可 以 不 必 遍 历数 据 表 中 的 所 有 记录 ， 而 是 查询 索引 列 。 一 般 过 程 的 数据 查询 是 通过 遍历 全 部 数据 ， 
并 寻找 数据 库 中 的 匹配 记录 而 实现 的 。 与 一 般 形式 的 查询 相 比 ， 索 引 就 像 一 本 书 的 目录 。 而 当 用 户 通 
过 目录 查找 书 中 内 容 时 ， 就 好 比 用 户 通 过 目录 查询 某 章节 的 某 个 知识 点 。 这 样 就 为 用 户 在 查找 内 容 过 
程 中 ， 缩 短 大 量 时 间 ， 帮 助 用 户 有 效 地 提高 查找 速度 。 所 以 ， 使 用 索引 可 以 有 效 地 提高 数据 库 系 统 的 
整体 性 能 。 

应 用 MySQL 数据 库 时 ， 并 非 用户 在 查询 数据 的 时 候 ， 总 需要 应 用 索引 来 优化 查询 。 凡 事 都 有 双 
面 性 ， 使 用 索引 可 以 提高 检索 数据 的 速度 ， 对 于 依赖 关系 的 子 表 和 父 表 之 问 的 联合 查询 时 ， 可 以 提高 
查询 速度 ， 并 且 可 以 提高 整体 的 系统 性 能 。 但 是 ， 创 建 索引 和 维护 需要 耗费 时 间 ， 并 且 该 耗费 时 间 与 
数据 量 的 大 小 成 正比 ， 另 外 ， 索 引 需 要 占用 物理 空间 ， 给 数据 的 维护 造成 很 多 麻烦 。 

整体 来 说 ， 索 引 可 以 提高 查询 的 速度 ， 但 是 会 影响 用 户 操作 数据 库 的 插入 操作 。 因 为 ， 向 有 索引 
的 表 中 插入 记录 时 ， 数 据 库 系 统 会 按照 索引 进行 排序 。 所 以 ， 用 户 可 以 将 索引 删除 后 ， 插 入 数据 ， 当 
数据 插入 操作 完成 后 ， 用 户 可 以 重新 创建 索引 。 


a 
0 培 明 
不 同 的 存储 引擎 定义 每 个 表 的 最 大 索引 数 和 最 大 索引 长 度 。 所 有 存储 引擎 对 每 个 表 至 少 支持 
16 个 索引 。 总 索引 长 度 至 少 为 256 字 节 。 有 些 存储 引擎 支持 更 多 的 索引 数 和 更 大 的 索引 长 度 。 索 
引 有 两 种 存储 类 型 ， 包 括 B 树 (BTREE ) 索引 和 哈 希 (HASH ) 索引 。 其 中 ，B 树 为 系统 默认 索引 
方法 。 


11.1.2 MySQL 索引 分 类 


MySQL 的 索引 包括 普通 索引 、 唯 一 性 索引 、 全 文 索 引 、 单 列 索 引 、 多 列 索引 和 空间 索引 等 。 


MySQL 从 入 门 到 精通 


1. 普通 索引 


普通 索引 ， 即 不 应 用 任何 限制 条 件 的 索引 ， 该 索引 可 以 在 任何 数据 类 型 中 创建 。 字 段 本 身 的 约束 
条 件 可 以 判断 其 值 是 否 为 空 或 唯一 。 创 建 该 类 型 索引 后 ， 用 户 在 查询 时 ， 便 可 以 通过 索引 进行 查询 。 
在 某 数据 表 的 某 一 字段 中 ， 建 立 普通 索引 后 。 用 户 需要 查询 数据 时 ， 只 需 根据 该 索引 进行 查询 即 可 。 


2. 唯一 性 索引 


使 用 UNIQUE 参数 可 以 设置 唯一 索引 。 创 建 该 索引 时 ， 索 引 的 值 必须 唯一 ， 通 过 唯一 索引 ， 用 户 
可 以 快速 定位 某 条 记录 ， 主 键 是 一 种 特殊 唯一 索引 。 


3. 全 文 索引 


使 用 FULLTEXT 参数 可 以 设置 索引 为 全 文 索引 。 全 文 索引 只 能 创建 在 CHAR、VARCHAR 或 者 
TEXT 类 型 的 字段 上 。 查 询 数据 量 较 大 的 字符 串 类 型 的 字段 时 ， 使 用 全 文 索引 可 以 提高 查询 速度 。 例 
如 ， 查 询 带 有 文章 回复 内 容 的 字段 ， 可 以 应 用 全 文 索引 方式 。 需 要 注意 的 是 ， 在 默认 情况 下 ， 应 用 全 
文 搜索 大 小 写 不 敏感 。 如 果 索 引 的 列 使 用 二 进 制 排序 后 ， 可 以 执行 大 小 写 敏感 的 全 文 索引 。 


4. 单列 索引 


顾名思义 ， 单 列 索引 即 只 对 应 一 个 字段 的 索引 。 其 可 以 包括 上 述 叙述 的 3 种 索引 方式 。 应 用 该 索 
引 的 条 件 只 需要 保证 该 索引 值 对 应 一 个 字段 即 可 。 


5. 多 列 索引 


多 列 索引 是 在 表 的 多 个 字段 上 创建 一 个 索引 。 该 索引 指向 创建 时 对 应 的 多 个 字段 ， 用 户 可 以 通过 
这 几 个 字段 进行 查询 。 要 想 应 用 该 索引 ， 用 户 必须 使 用 这 些 字 段 中 的 第 一 个 字段 。 


6. 空间 索引 


使 用 SPATIAL 参数 可 以 设置 索引 为 空间 索引 。 空 间 索 引 只 能 建立 在 空间 数据 类 型 上 ， 这 样 可 以 提高 
系统 获取 空间 数据 的 效率 。 MySQL 中 只 有 MyISAM 存储 引擎 支持 空间 检索 , 而 且 索 引 的 字段 不 能 为 空 值 。 


11.2 创建 索引 


创建 索引 是 指 在 某 个 表 中 至 少 一 列 中 建立 索引 ， 以 便 提高 数据 库 性 能 。 其 中 ， 建 立 索引 可 以 提高 
表 的 访问 速度 。 本 节 通过 几 种 不 同 的 方式 创建 索引 。 其 中 包括 在 建立 数据 库 时 创建 索引 、 在 已 经 建立 
的 数据 表 中 创建 索引 和 修改 数据 表 结构 创建 索引 。 

11.2.1 在 建立 数据 表 时 创建 索引 


在 建立 数据 表 时 可 以 直接 创建 索引 ， 这 种 方式 比较 直接 ， 且 方便 、 易 用 。 在 建立 数据 表 时 创建 索 


引 的 基本 语法 结构 如 下 。 


create table table_name( 
属性 名 数据 类 型 [约束 条 件 ]， 
属性 名 数据 类 型 [约束 条 件 ] 


属性 名 数据 类 型 

[UNIQUE | FULLTEXT | SPATIAL] INDEX }KEY 
[别名 1( 属性 名 1[( 长 度 )] [ASC | DESC]) 

下 


其 中 ， 属 性 名 后 的 属性 值 ， 其 含义 如 下 。 

(1) UNIQUE: 可 选项 ， 表 明 索 引 为 唯一 性 索引 。 

(2) FULLTEXT: 可 选项 ， 表 明 索 引 为 全 

(3) SPATIAL: 可 选项 ， 表 明 索 引 为 空间 索 

INDEX 和 KEY 参数 用 于 指定 字段 索引 ， 用 户 在 选择 时 ， 只 需要 选择 其 中 的 一 种 即 可 ， 另 外 别名 
为 可 选项 ， 其 作用 是 给 创建 的 索引 取 新 名 称 ， 别 名 的 参数 

(1) 属性 名 1: 指 索 引 对 应 的 字段 名 称 ， 该 字段 必须 

(2) 长 度 : 可 选项 ， 指 索引 的 长 度 ， 必 须 是 字符 串 类 型 才 可 以 使 用 。 

(3) ASC/DESC: 可 选项 ，ASC 表示 升序 排列 ，DESC 参数 表示 降序 排列 。 

1. 普通 索引 创建 


创建 普通 索引 ， 即 不 添加 UNIQUE、FULLTEXT 等 任何 参数 。 
例 11.1 下 面 创建 表 名 为 score 的 数据 表 , 并 在 该 表 的 id 字段 上 建立 索引 ,其 主要 代码 如 下 。( 实 
例 位 置 ， 光盘 \TMNsNIL11.1) 


create table score( 

id int(11) auto_increment primary key not null, 
name varchar(50) not null, 

math int(5) not null, 

english int(5) not null, 

chinese int(5) not null, 

index(id)); 
















运行 以 上 代码 的 结果 如 图 11.1 所 示 。 


> create table score( 
to_increment prinary key not null,. 
> not null. 


affected (0.08 sec)? 

















11.1 创建 普通 索引 
在 命令 提示 符 中 使 用 SHOW CREATE TABLE 语句 查看 该 表 的 结构 ， 在 命令 提示 符 中 输入 的 代码 
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如 下 。 
show create table score; 


其 运行 结果 如 图 11.2 所 示 。 





图 11.2 查看 数据 表 结 构 
从 图 11.2 中 可 以 清晰 地 看 到 ， 该 表 结构 的 索引 为 4， 则 可 以 说 明 该 表 的 索引 建立 成 功 。 
2. 创建 唯一 性 索引 





创建 唯一 性 索引 与 创建 一 般 索 引 的 语法 结构 大 体 相 同 ， 但 是 在 创建 唯一 索引 的 时 候 ， 需 要 使 用 
UNIQUE 参数 进行 约束 。 

例 11.2 创建 一 个 表 名 为 address 的 数据 表 ， 并 指定 该 表 的 id 字段 上 建立 唯一 索引 ， 其 代码 如 下 
所 示 。 (实例 位 置 ， 光盘 \TMNsNM1\11.2) 


create table address( 

id int(11) auto_increment primary key not null, 
name varchar(50), 

address varchar(200), 

UNIQUE INDEX address(id ASC)); 


应 用 SHOW CREATE TABLE 语句 查看 表 的 结构 ， 其 运行 如 图 11.3 所 示 。 





图 11.3 查看 唯一 索引 的 表 结 构 


从 图 11.3 中 可 以 看 到 ， 该 表 的 id 字段 上 已 经 建立 了 一 个 名 为 address 的 唯一 索引 。 
4 
Ms 说明 
虽然 添加 唯一 索引 可 以 约束 字段 的 唯一 性 , 但 是 有 时 候 并 不 能 提高 用 户 查找 速度 ， 即 不 能 实现 
优化 查询 目的 。 所 以 ， 读 者 在 使 用 过 程 中 需要 根据 实际 情况 来 选择 唯一 索引 。 
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3. 创建 全 文 索引 

与 创建 普通 索引 和 唯一 索引 不 同 ,全文 索 引 的 创建 只 能 作用 在 CHAR、VARCHAR、TEXT 类 型 的 
字段 上 。 创 建 全文 索 引 需 要 使 用 FULLTEXT 参数 进行 约束 。 

例 11.3 创建 一 个 名 称 为 cards 的 数据 表 , 并 在 该 表 的 number 字段 上 创建 全 文 索引 , 其 代码 如 下 。 
(实例 位 置 ， 光盘 \TMNsIM1\11.3) 





create table cards( 

id int(11) auto_increment primary key not null, 
name varchar(50), 

number bigint(11), 

info varchar(50), 

FULLTEXT KEY cards_info(info)) engine=MylSAM.; 


在 命令 提示 符 中 应 用 SHOW CREATE TABLE 语句 查看 表 结构 ， 其 代码 如 下 。 
SHOW CREATE TABLE cards; 
运行 结果 如 图 11.4 所 示 。 

REATE TABLE “ 


1> NOT NUL 
58> default NU 


am 
> ENGINE=MyISAM DEFRULT 





图 11.4 查看 全 文 索引 的 数据 表 结 构 
Ci 蛆 
只 有 MyISAM 类 型 的 数据 表 支 持 FULLTEXT 全 文 索引 ，InnoDB 或 其 他 类 型 的 数据 表 不 支持 
全 文 索 引 。 当 用 户 在 建立 全 文 索引 的 时 候 ， 返回 “ERROR 1283 (HY000): Column number cannot be 
part of FULLTEXT index” 的 错误 , 则 说 明 用 户 操 作 的 当前 数据 表 不 支持 全 文 索 引 , 即 不 为 MYISAM 
类 型 的 数据 表 。 


4. 创建 单列 索引 

创建 单列 索引 ， 即 在 数据 表 的 单个 字段 上 创建 索引 。 创 建 该 类 型 索引 不 需要 引入 约束 参数 ， 用 户 
在 建立 时 只 需 指定 单列 字段 名 ， 即 可 创建 单列 索引 。 

例 11.4 创建 名 称 为 telephone 的 数据 表 ， 并 指定 在 tel 字段 上 建立 名 称 为 tel_num 的 单列 索引 ， 
其 代码 如 下 。《 实 例 位 置 : 光盘 \TMsNII11.4) 

create table telephone( 


id int(11) primary key auto_increment not null, 
name varchar(50) not null, 


1 


MySQL 从 入 门 到 精通 
tel varchar(50) not null, 
index tel_num(tel(20)) 
); 


运行 上 述 代码 后 ， 应 用 SHOW CREATE TABLE 语句 查看 表 的 结构 ， 其 运行 结果 如 图 11.5 所 示 。 


PRIMARY KEY 


KEY “tel num ¢ 
》 ENGINE 





图 11.5 查看 单列 索引 表 的 数据 表 结构 
ox、 
说 明 


数据 表 中 的 字段 长 度 为 50， 而 创建 的 索引 的 字段 长 度 为 20， 这 样 做 的 目的 是 为 了 提高 查询 效 
率 ， 优 化 查询 速度 。 


5. 创建 多 列 索引 


与 创建 单列 索引 相仿 ， 创 建 多 列 索 引 即 指定 表 的 多 个 字段 即 可 实现 。 
例 11.5 创建 名 称 为 information 的 数据 表 ， 并 指定 name 和 sex 为 多 列 索 引 ， 其 代码 如 下 。〔 实 
例 位 置 ， 光盘 \TMNsIMI1\11.5) 


create table information( 

id int(11) auto_increment primary key not null, 
name varchar(50) not null, 

sex varchar(5) not null, 

birthday varchar(50) not null, 

INDEX info(name,sex) 

js 


应 用 SHOW CREATE TABLE 语句 查看 创建 多 列 的 数据 表 结 构 ， 其 运行 结果 如 图 11.6 所 示 。 


informat ion* 
T NULL auto_increment- 





11.6 查看 多 列 索 引 表 的 数据 结构 


需要 注意 的 是 ， 在 多 列 索引 中 ， 只 有 查询 条 件 中 使 用 了 这 些 字 段 中 的 第 一 个 字段 〈 即 上 面 示例 中 
的 name 字段 ) 时 ， 索 引 才 会 被 使 用 。 
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wa 
说 明 
触发 多 列 索 引 的 条 件 是 用 户 必须 使 用 索引 的 第 一 字段 ， 如果 没有 用 到 第 一 字段 ， 则 索引 不 起 任 
何 作 用 ， 用 户 想 要 优化 查询 速度 ， 可 以 应 用 该 类 索引 形式 。 


6. 创建 空间 索引 


创建 空间 索引 时 ， 需 要 设置 SPATIAL 参数 。 同 样 ， 必 须 说 明 的 是 ， 只 有 MyISAM 类 型 表 支 持 该 
类 型 索引 。 而 且 ， 索 引 字段 必须 有 非 空 约束 。 

例 11.6 创建 一 个 名 称 为 list 的 数据 表 ， 并 创建 一 个 名 为 listinfo 的 空间 索引 ， 其 代码 如 下 。〔 实 
例 位 置 ， 光盘 \TMNsN1L11.6) 


create table list( 

id int(11) primary key auto_increment not null, 
goods geometry not null, 

SPATIAL INDEX listinfo(goods) 
)engine=MylSAM; 


运行 上 述 代码 ， 创 建成 功 后 ， 在 命令 提示 符 中 应 用 SHOW CREATE TABLE 语句 查看 表 的 结构 。 
其 运行 结果 如 图 11.7 所 示 。 





图 11.7 查看 空间 索引 表 的 结构 
从 图 11.7 中 可 以 看 到 ，goods 字段 上 已 经 建立 名 称 为 listinfo 的 空间 索引 ， 其 中 ，goods 字段 必须 








不 能 为 空 ， 且 数据 类 型 是 GEOMETRY。 该 类 型 是 空间 数据 类 型 。 空 间 类 型 不 能 用 其 他 类 型 代 蔡 ， 否 
则 在 生成 空间 索引 时 会 产生 错误 且 不 能 正常 创建 该 类 型 索引 。 


VT 
室 间 类 型 除了 上 述 示 例 中 提 到 的 GEOMETRY 类 型 外 ， 还 包括 如 POINT、LINESTRING、 
POLYGON 等 类 型 。 这 些 空间 数据 类 型 在 平常 的 操作 中 很 少 被 用 到 。 


11.2.2 在 已 建立 的 数据 表 中 创建 索引 

















在 MySQL 中 ， 不 但 可 以 在 用 户 创建 数据 表 时 创建 索引 ， 用 户 也 可 以 直接 在 已 经 创建 的 表 中 ， 在 
已 经 存在 的 一 个 或 几 个 字段 上 创建 索引 。 其 基本 的 命令 结构 如 下 所 示 。 


CREATE [UNIQUE | FULLTEXT |SPATIAL ] INDEX index_name 
ON table_name( 属 性 [(length)] [ ASC | DESC]); 
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命令 的 参数 说 明 如 下 。 


(1) index_name 为 索引 名 称 ， 该 参数 作用 是 给 用 户 创建 的 索引 赋予 新 的 名 称 。 
(2) table name 为 表 名 ， 即 指定 创建 索引 的 表 名 称 。 
(3) 可 选 参数 ,指定 索引 类 型 , 包括 UNIQUE (唯一 索引 ) 
(空间 索引 ) 。 
(4) 属性 参数 ， 指 定 索引 对 应 的 字段 名 称 。 该 字段 必须 已 经 预存 在 用 户 想 要 操作 的 数据 表 中 ， 如 
果 该 数据 表 中 不 存在 用 户 指定 的 字段 ， 则 系统 会 提示 异常 。 
(5) length 为 可 选 参数 ， 用 于 指定 索引 长 度 。 
(6) ASC 和 DESC 参数 ， 指 定数 据 表 的 排序 顺序 。 
与 建立 数据 表 时 创建 索引 相同 ， 在 已 建立 的 数据 表 中 创建 索引 同样 包含 6 种 索引 方式 。 
1. 创建 普通 索引 


例 11.7 首先 ,应 用 SHOW CREATE TABLE 语句 查看 studentinfo 表 的 结构 , 其 运行 结果 如 图 11.8 
所 示 。《〔 实 例 位 置 ， 光盘 \TMNsIM1\11.7) 


、FULLTEXT (全 文 索引 ) 、SPATIAL 








图 11.8 查看 未 添加 索引 前 的 表 结 构 
然后 ， 在 该 表 中 创建 名 称 为 stu_info 的 普通 索引 ， 
create INDEX stu_info ON studentinfo(sid); 
输入 上 述 命令 后 ， 

所 示 。 


在 命令 提示 符 中 输入 如 下 命令 。 


应 用 SHOW CREATE TABLE 语句 查看 该 数据 表 的 结构 。 其 运行 结果 如 图 11.9 


CREATE INBLE 
NULL a 


MENT 19 DEFAULT ¢ 





图 11.9 查看 添加 索引 后 的 表格 结构 


从 图 11.9 中 可 以 看 出 ， 名 称 为 stu_info 的 数据 表 创 建成 功 。 如 果 系 统 没有 提示 异常 或 错误 ， 则 说 
明 已 经 向 studentinfo 数据 表 中 建立 名 称 为 stu_info 的 普通 索引 。 


2. 创建 唯一 索引 
在 已 经 存在 的 数据 表 中 建立 唯一 索引 的 命令 如 下 。 
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CREATE UNIQUE INDEX 索引 名 ON 数据 表 名 称 (字段 名 称 ); 
其 中 ，UNIQUE 是 用 来 设置 索引 唯一 性 的 参数 ， 该 表 中 的 字段 名 称 既 可 以 存在 唯一 性 约束 ， 也 可 
以 不 存在 唯一 性 约束 。 


例 11.8 下 面 在 indexl 表 中 的 cid 字段 上 建立 名 为 index1_id 的 唯一 性 索引 ， 代 码 如 下 。 (实例 位 
置 : 光盘 \TMNsN11\11.8) 


CREATE UNIQUE INDEX index1_id ON index1(cid); 








输入 上 述 命令 后 ,应 用 SHOW CREATE TABLE 语句 查看 该 数据 表 的 结构 。 其 运行 结果 如 图 11.10 
所 示 。 


IT NULL- 
NULL, 


NOT NULL. 


PRIMARY 
UNI QUE 


> ENCINE LT CHA utf8 COLLATE-utf8_u 


<@.00 sec)> 





图 11.10 查看 添加 唯一 索引 后 的 表格 结构 
3. 创建 全 文 索引 
在 MySQL 中 ， 为 已 经 存在 的 数据 表 创 建 全 文 索引 的 命令 如 下 。 
CREATE FULLTEXT INDEX 索引 名 ON 数据 表 名 称 (字段 名 称 ); 
其 中 , FULLTEXT 用 来 设置 索引 为 全 文 索引 。 操作 的 数据 表 类 型 必须 为 MyISAM 类 型 。 字 段 类 型 
必须 为 VARCHAR、CHAR、TEXT 等 类 型 。 


例 11.9 下 面 在 index2 表 中 的 info 字段 上 建立 名 为 index2_info 的 全 文 索引 ， 代 码 如 下 。〔 实 例 
位 置 : 光盘 \TMNsINLLN11.9) 


CREATE FULLTEXT INDEX index2_info ON index2(info); 


输入 上 述 命 令 后 ， 应 用 SHOW CREATE TABLE 语句 查看 该 数据 表 的 结构 。 其 运行 结果 如 图 11.11 
所 示 。 





图 11.11 查看 添加 全 文 索引 后 的 表格 结构 
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4. 创建 单列 索引 

与 建立 数据 表 时 创建 单列 索引 相同 ， 用 户 可 以 设置 单列 索引 。 其 命令 结构 如 下 。 

CREATE INDEX 索引 名 ON 数据 表 名 称 (字段 名 称 (长 度 )); 

设置 字段 名 称 长 度 ， 可 以 优化 查询 速度 ， 提 高 查询 效率 。 

例 11.10 下 面 在 index3 表 中 的 address 字段 上 建立 名 为 index3_addr 的 单列 索引 。Address 字段 的 
数据 类 型 为 varchar(20)， 索 引 的 数据 类 型 为 char(4)， 代 码 如 下 。 《实例 位 置 ; 光盘 \TMNsNIL11.10) 

CREATE INDEX index3_addr ON index3(address(4)); 

输入 上 述 命令 后 ， 应 用 SHOW CREATE TABLE 语句 查看 该 数据 表 的 结构 。 其 运行 结果 如 图 11.12 
所 示 。 

日 index3 ! a TABLE ‘index3 


* intC11> NOT NULL AUTO_INCREMENT, 
ATE utf8_unicode ci NOT NULL, 


“C4)) 
SET=utf8 COLLATE=utf8_unicode_ci ! 





图 11.12 查看 添加 单列 索引 后 的 表格 结构 

5. 创建 多 列 索引 

建立 多 列 索引 与 建立 单列 索引 类 似 。 其 主要 命令 结构 如 下 。 

CREATE INDEX 索引 名 ON 数据 表 名 称 (字段 名 称 1, 字 段 名 称 2,…); 

与 建立 数据 表 时 创建 多 列 索引 相同 ， 当 创建 多 列 索 引 时 ， 用 户 必须 使 用 第 一 字段 作为 查询 条 件 ， 
否则 索引 不 能 生效 。 

例 11.11 下 面 在 index4 表 中 的 name 和 address 字段 上 建立 名 为 index4_na 的 多 列 索 引 ， 代 码 如 
下 。 《实例 位 置 : 光盘 \TMNsIM1\11.11) 


CREATE INDEX index4_na ON index4(name,address); 


输入 上 述 命令 后 ,， 应 用 SHOW CREATE TABLE 语句 查看 该 数据 表 的 结构 。 其 运行 结果 如 图 11.13 
所 示 。 


MENT - 


NULL, 
ci NOT NULL, 





图 11.13 查看 添加 多 列 索引 后 的 表格 结构 
6. 创建 空间 索引 
建立 空间 索引 ， 用 户 需 要 应 用 SPATIAL 参数 作为 约束 条 件 。 其 命令 结构 如 下 。 
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CREATE SPATIAL INDEX 索引 名 ON 数据 表 名 称 (字段 名 称 ); 


其 中 ，SPATIAL 用 来 设置 索引 为 空间 索引 。 用 户 要 操作 的 数据 表 类 型 必须 为 MyISAM 类 型 。 并且 
字段 名 称 必须 存在 非 空 约束 ， 否 则 将 不 能 正常 创建 空间 索引 。 





11.2.3 ”修改 数据 表 结构 添加 索引 


修改 已 经 存在 表 上 的 索引 ， 可 以 通过 ALTER TABLE 语句 为 数据 表 添 加 索引 ， 其 基本 结构 如 下 。 


ALTER TABLE table_name ADD [ UNIQUE | FULLTEXT |SPATIAL ] INDEX index_name( 属 性 名 [(length)] 
[ASC | DESC]): 


该 参数 与 11.2.1 节 和 11.2.2 节 中 所 介绍 的 参数 相同 ,这 里 不 再 袭 述 , 请 读者 参阅 前 面 两 节 中 的 内 容 。 
.添加 普通 索引 
首先 ， 应 用 SHOW CREATE TABLE 语句 查看 studentinfo 表 的 结构 ， 其 运行 结果 如 图 11.14 所 示 。 


CREMENT =18 DEF 





图 11.14 查看 未 添加 索引 前 的 表 结 构 
然后 ， 在 该 表 中 添加 名 称 为 timer 的 普通 索引 ， 在 命令 提示 符 中 输入 如 下 命令 。 
alter table studentinfo ADD INDEX timer (time(20)); 


输入 上 述 命令 后 ,应 用 SHOW CREATE TABLE 语句 查看 该 数据 表 的 结构 。 其 运行 结果 如 图 11.15 
所 示 。 








图 11.15 查看 添加 索引 后 的 表格 结构 


从 图 11.15 中 可 以 看 出 ， 名 称 为 timer 的 数据 表 添加 成 功 ， 己 经 成 功 向 studentinfo 数据 表 中 添加 名 
称 为 timer 的 普通 索引 。 
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从 功能 上 看 ,修改 数据 表 结 构 添 加 索引 与 在 已 存在 数据 表 中 建立 索引 所 实现 功能 大 体 相 同 ， 二 
者 均 是 在 已 经 建立 的 数据 表 中 添加 或 创建 新 的 索引 。 所以, 用 户 在 使 用 的 时 候 ,， 可 以 根据 个 人 需求 
和 实际 情况 ， 选 择 适合 的 方式 向 数据 表 中 添加 索引 。 


2. 添加 唯一 索引 
与 已 存在 的 数据 表 
ALTER TABLE 表 名 


其 中 ， ALTER 语句 一 般 是 用 来 修改 数据 表 结 构 的 语句 ，ADD 为 添加 索引 的 关 


用 来 设置 索引 唯一 性 的 


TEXT 类 型 的 字段 上 。 创 建 全 


文 索引 。 


3. 添加 全 文 索引 








中 添加 索引 的 过 程 类 似 ， 在 数据 表 中 添加 唯一 索引 的 命令 结构 如 下 所 示 。 
ADD UNIQUE INDEX 索引 名 称 *( 字 段 名 称 ); 


字 ; UNIQUE 是 
参数 ， 该 表 中 的 字段 名 称 既 可 以 存在 唯一 性 约束 ， 也 可 以 不 存在 唯一 性 约束 。 





创建 全 文 索 引 与 创建 普通 索引 和 唯一 索引 不 同 ， 全 文 索引 创建 只 能 作用 在 CHAR、VARCHAR、 





文 索引 需要 使 用 FULLTEXT 参数 进行 约束 。 


在 MySQL 中 ， 为 已 经 存在 的 数据 表 添 加 全 文 索 引 的 命令 如 下 。 
ALTER TABLE 表 名 ADD FULLTEXT INDEX 索引 名 称 (字段 名 称 ); 


其 中 ，ADD 是 添加 的 关键 字 ，FULLTEXT 用 来 设置 索引 为 全 文 索引 。 操 作 的 数据 表 类 型 必须 为 
MyISAM 类 型 。 字 段 类 型 同样 必须 为 VARCHAR、CHAR、TEXT 等 类 型 。 
例 11.12 使 用 ALTER INDEX 语句 在 数据 表 workinfo 的 address 字段 上 创建 名 为 index_ext 的 全 


用 修改 数据 表 结 果 


(实例 位 置 ， 光盘 \TMNsMN1\11.12) 





方式 添加 全 文 索引 。 用 ALTER INDEX 语句 在 address 字段 上 创建 名 为 


index_ext 的 全 文 索引 ， 具 体 代码 如 下 。 
ALTER TABLE workinfo ADD FULLTEXT INDEX index_ext(address); 
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输入 上 述 命令 后 ,应 用 SHOW CREATE TABLE 语句 查看 该 数据 表 的 结构 。 其 运行 结果 如 图 11.16 
所 示 。 


4. 添加 单列 索引 





图 11.16 查看 使 用 ALTER TABLE 语句 创建 的 全 文 索引 


与 建立 数据 表 时 创建 单列 索引 相同 ， 用 户 可 以 设置 单列 索引 。 其 命令 结构 如 下 。 
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ALTER TABLE 表 名 ADD INDEX 索引 名 称 (字段 名 称 (长 度 )); 

同样 ， 用 户 可 以 设置 字段 名 称 长 度 ， 以 便 优化 查询 速度 ， 提 高 执行 效率 。 

5. 添加 多 列 索引 

添加 多 列 索引 与 建立 单列 索引 类 似 。 其 主要 命令 结构 如 下 。 

ALTER TABLE 表 名 ADD INDEX 索引 名 称 (字段 名 称 1, 字 段 名 称 2,…); 

使 用 ALTER 修改 数据 表 结 构 同 样 可 以 添加 多 列 索引 。 与 建立 数据 表 时 创建 多 列 索引 相同 ， 当 创建 
多 列 索引 时 ， 用 户 必须 使 用 第 一 字段 作为 查询 条 件 ， 否 则 索引 不 能 生效 。 

6. 添加 空间 索引 

添加 空间 索引 ， 用 户 需要 应 用 SPATIAL 参数 作为 约束 条 件 。 其 命令 结构 如 下 。 

ALTER TABLE 表 名 ADD SPATIAL INDEX 索引 名 称 (字段 名 称 ); 

其 中 ，SPATIAL 用 来 设置 索引 为 空间 索引 。 用 户 要 操作 的 数据 表 类 型 必须 为 MyISAM 类 型 ， 并 且 
字段 名 称 必须 存在 非 空 约 束 ， 否 则 将 不 能 正常 创建 空间 索引 。 该 类 别 索 引 并 不 常用 ， 初 学 者 只 需要 了 
解 该 索引 类 型 即 可 。 





11.3 删除 索引 


在 MySQL 中 ， 创 建 索引 后 ， 如 果 用 户 不 再 需要 该 索引 ， 则 可 以 删除 指定 表 的 索引 。 因 为 这 些 已 
经 被 建立 且 不 常 使 用 的 索引 ， 一 方面 可 能 会 占用 系统 资源 ， 另 一 方面 也 可 能 导致 更 新 速度 下 降 ， 这 极 
大 地 影响 了 数据 表 的 性 能 。 所 以 ， 在 用 户 不 需要 该 表 的 索引 时 ， 可 以 手动 删除 指定 索引 。 其 中 ， 删 除 
索引 可 以 通过 DROP 语句 来 实现 。 其 基本 的 命令 如 下 。 

DROP INDEX index_name ON table_name; 


其 中 ， 参 数 index_name 是 用 户 需要 删除 的 索引 名 称 ， 参 数 table_name 指定 数据 表 名 称 ， 下 面 应 用 
示例 向 读者 展示 如 何 删除 数据 表 中 已 经 存在 的 索引 。 打 开 MySQL 后 ， 应 用 SHOW CREATE TABLE 
语句 查看 数据 表 的 索引 ， 其 运行 结果 如 图 11.17 所 示 。 








图 11.17 查看 address 数据 表 内 的 索引 





从 图 11.17 中 可 以 看 出 ， 名 称 为 address 的 数据 表 中 存在 唯一 索引 address。 在 命令 提示 符 中 继续 输 
入 如 下 命令 。 
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DROP INDEX id ON address 

运行 上 述 代码 的 结果 如 图 11.18 所 示 。 

在 用 户 顺 利 删除 索引 后 , 为 确定 该 索引 是 否 已 被 删除 , 用 户 可 以 再 次 应 用 SHOW CREATE TABLE 
语句 来 查看 数据 表 结 构 。 其 运行 结果 如 图 11.19 所 示 。 





图 11.18 删除 唯一 索引 address 图 11.19 再 次 查看 address 数据 表 结 构 


从 图 11.19 可 以 看 出 ， 名 称 为 address 的 唯一 索引 已 经 被 删除 。 
例 11.13 本 实例 将 使 用 DROP 语句 从 数据 表 中 删除 不 再 需要 的 索引 ,效果 如 图 11.20 所 示 。 


例 位 置 ， 光盘 \TMNsINIL11.13) 





迪 


1> DROP INDE 
OK。B 


ds: 8 Duplica 





图 11.20 删除 唯一 性 索引 
使 用 DROP 语句 删除 workinfo 表 的 唯一 性 索引 index_id， 具 体 代 码 如 下 。 
DROP INDEX index_id ON workinfo; 


11.4 小 结 


本 章 对 MySQL 数据 库 的 索引 的 基础 知识 、 创 建 索引 、 删 除 索引 进行 了 详细 讲解 ， 创 建 索 引 的 内 
容 是 本 章 的 重点 。 读 者 应 该 重点 掌握 创建 索引 的 3 种 方法 , 分 别 为 创建 表 的 时 候 创 建 索引 、 使 用 CREATE 
INDEX 语句 来 创建 索引 和 使 用 ALTER TABLE 语句 来 创建 索引 。 


11.5 实践 与 练习 


1. 运用 CREATE INDEX 语句 为 name 字段 创建 长 度 为 10 的 索引 index_name。 (答案 位 置 : 光盘 \ 
TMsNLN11.14) 

2. 创建 一 个 表 名 为 tb_user 的 数据 表 ， 并 在 该 表 的 id 字段 上 建立 唯一 索引 。 (答案 位 置 ， 光 盘 \ 
TMsNLN11.15) 
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视 图 


( a" 视频 讲解 .22 分 钟 ) 


视图 是 从 一 个 或 多 个 表 中 导出 的 表 , 是 一 种 虚拟 存在 的 表 。 视 图 就 像 一 个 窗口 ， 
通过 这 个 窗口 可 以 看 到 系统 专门 提供 的 数据 。 这 样 ， 用 户 可 以 不 用 看 到 整个 数据 库 
表 中 的 数据 ， 而 只 关心 对 自己 有 用 的 数据 。 视 图 可 以 使 用 户 的 操作 更 方便 ， 而 且 可 
以 保障 数据 库 系统 的 安全 性 。 本 章 将 介绍 视图 的 含义 和 作用 ， 视 图 定义 的 原则 和 创 
建 视图 的 方法 ， 并 对 修改 视图 、 坦 看 视图 和 删除 视图 的 方法 进行 了 详细 的 讲解 。 
通过 阅读 本 章 ， 读 者 可 以 : 
了 解 使 用 CREATE VIEW 语句 创建 视图 的 方法 
了 解 创建 视图 的 注意 事项 
掌握 使 用 SHOW TABLE STATUS 语句 查看 视图 的 方法 
掌握 使 用 CREATE OR REPLACE VIEW 语句 修改 视图 的 方法 
掌握 使 用 ALTER 语句 修改 视图 的 方法 
掌握 更 新 视图 和 使 用 DROP VIEW 语句 删除 视图 的 方法 


各 吾 吾 吾 至 至 
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12.1 视图 概述 


视图 是 由 数据 库 中 的 一 个 表 或 多 个 表 导 出 的 虚拟 表 ， 方 便 用 户 对 数据 的 操作 。 本 节 将 详细 讲解 视 
图 的 概念 及 作用 。 


12.1.1 视图 的 概念 


视图 是 一 个 虚拟 表 ， 是 从 数据 库 中 一 个 或 多 个 表 中 导出 来 的 表 ， 其 内 容 由 查询 定义 。 同 真实 的 表 
一 样 ， 视 图 包含 一 系列 带 有 名 称 的 列 和 行 数据 。 但 是 ， 数 据 库 中 只 存放 了 视图 的 定义 ， 而 并 没有 存放 
视图 中 的 数据 。 这 些 数 据 存放 在 原来 的 表 中 。 使 用 视图 查询 数据 时 ， 数 据 库 系 统 会 从 原来 的 表 中 取出 
对 应 的 数据 。 因 此 ， 视 图 中 的 数据 是 依赖 于 原来 的 表 中 的 数据 的 。 一 旦 表 中 的 数据 发 生 改 变 ， 显 示 在 
视图 中 的 数据 也 会 发 生 改 变 。 

视图 是 存储 在 数据 库 中 的 查询 的 SQL 语句 ， 它 主要 出 于 两 种 原因 : 安全 原因 ， 视 图 可 以 隐藏 一 些 
数据 ， 例 如 ， 员 工 信 息 表 ， 可 以 用 视图 只 显示 姓名 、 工 龄 、 地 址 ， 而 不 显示 社会 保险 号 和 工资 数 等 
另 一 个 原因 是 可 使 复杂 的 查询 易于 理解 和 使 用 。 


12.1.2 ”视图 的 作用 





对 其 中 所 引用 的 基础 表 来 说 ， 视 图 的 作用 类 似 于 筛选 。 定 义 视 图 的 筛选 可 以 来 自 当 前 或 其 他 数据 
库 的 一 个 或 多 个 表 ， 或 者 其 他 视图 。 通 过 视图 进行 查询 没有 任何 限制 ， 通 过 它们 进行 数据 修改 时 的 限 
制 也 很 少 。 下 面 将 视图 的 作用 归纳 为 如 下 几 点 。 

1. 简单 性 

看 到 的 就 是 需要 的 。 视 图 不 仅 可 以 简化 用 户 对 数据 的 理解 ， 也 可 以 简化 他 们 的 操作 。 那 些 被 经 常 
使 用 的 查询 可 以 被 定义 为 视图 ， 从 而 使 得 用 户 不 必 为 以 后 的 操作 每 次 指定 全 部 的 条 件 。 


2. 安全 性 
视图 的 安全 性 可 以 防止 未 授权 用 户 查 看 特定 的 行 或 列 ， 使 有 权限 用 户 只 能 看 到 表 中 特定 行 的 方法 
如 下 。 


(1) 在 表 中 增加 一 个 标志 用 户 名 的 列 。 

(2) 建立 视图 ， 使 用 户 只 能 看 到 标 有 自己 用 户 名 的 行 。 
(3) 把 视图 授权 给 其 他 用 户 。 

3. 逻辑 数据 独立 性 


视图 可 以 使 应 用 程序 和 数据 库 表 在 一 定 程度 上 独立 。 如 果 没有 视图 ， 程 序 一 定 是 建立 在 表 上 的 。 
有 了 视图 之 后 ， 程 序 可 以 建立 在 视图 之 上 ， 从 而 程序 与 数据 库 表 被 视图 分 割 开 来 。 视 图 可 以 在 以 下 几 
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个 方面 使 程序 与 数据 独立 。 

(1) 如 果 应 用 建立 在 数据 库 表 上 ， 当 数据 库 表 发 生变 化 时 ， 可 以 在 表 上 建立 视图 ， 通 过 视图 屏蔽 
表 的 变化 ， 从 而 使 应 用 程序 可 以 不 动 。 

(2) 如 果 应 用 建立 在 数据 库 表 上 ， 当 应 用 发 生变 化 时 ， 可 以 在 表 上 建立 视图 ， 通过 视图 屏蔽 应 用 
的 变化 ， 从 而 使 数据 库 表 不 动 。 

(3) 如 果 应 用 建立 在 视图 上 ， 当 数据 库 表 发 生变 化 时 ， 可 以 在 表 上 修改 视图 ， 通 过 视图 屏蔽 表 的 
变化 ， 从 而 使 应 用 程序 可 以 不 动 。 

(4) 如 果 应 用 建立 在 视图 上 ， 当 应 用 发 生变 化 时 ， 可 以 在 表 上 修改 视图 ， 通 过 视图 屏蔽 应 用 的 变 
化 ， 从 而 使 数据 库 可 以 不 动 。 


























12.2 创建 视图 


创建 视图 是 指 在 已 经 存在 的 数据 库 表 上 建立 视图 ， 视 图 可 以 建立 在 一 张 表 中 ， 也 可 以 建立 在 多 张 
表 中 。 本 节 主 要 讲解 创建 视图 的 方法 。 


12.2.1 查看 创建 视图 的 权限 


创建 视图 需要 具有 CREATE VIEW 的 权限 ， 同 时 应 该 具有 查询 涉及 的 列 的 SELECT 权限 。 可 以 使 
用 SELECT 语句 来 查询 这 些 权限 信息 ， 查 询 语法 如 下 。 
SELECT Selete_priv,Create_view_priv FROM mysql.user WHERE user=' 用 户 名 '; 


(1) Selete_priv 属性 表示 用 户 是 否 具有 SELECT 权限 ，Y 表示 拥有 SELECT 权限 ，N 表示 没有 。 

(2) Create_view_priv 属性 表示 用 户 是 否 具有 CREATE VIEW 权限 ; mysql.user 表示 MySQL 数据 
库 下 面 的 user 表 。 

(3) “用 户 名 ”参数 表示 要 查询 是 否 拥有 DROP 权限 的 用 户 ， 该 参数 需要 用 单 引 号 引起 来 。 

例 12.1 下 面 查询 MySQL 中 root 用 户 是 否 具 有 创建 视图 的 权限 ， 代 码 如 下 。 “实例 位 置 ， 光盘 
\TMNsNM2\12.1) 

SELECT Select_priv,Create_view_priv FROM mysql.user WHERE user='root'; 


执行 结果 如 图 12.1 所 示 。 





et 《9.68 sec? 


图 12.1 查看 用 户 是 否 具 有 创建 视图 的 权限 

















结果 中 Select_ priv 和 Create_view_priv 属性 的 值 都 为 Y, 这 表示 root 用 户 具 有 SELECT 和 CREATE 
VIEW 权限 。 
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12.2.2 ”创建 视图 的 步骤 





MySQL 中 ， 创 建 视图 是 通过 CREATE VIEW 语句 实现 的 ， 其 语法 如 下 。 


CREATE [ALGORITHM={UNDEFINEDIMERGEITEMPTABLEj 
VIEW 视图 名 [( 属 性 清单 ) 
AS SELECT 语句 
[WITH [CASCADEDILOCAL] CHECK OPTION]:; 


(1) ALGORITHM 是 可 选 参数 ， 表 示 视 图 选择 的 算法 ; 
(2) “视图 名 ”参数 表示 要 创建 的 视图 名 称 ; 
(3) “属性 清单 ”是 可 选 参数 ， 指 定 视图 中 各 个 属性 的 名 词 ， 默 认 情况 下 与 SELECT 语句 中 查询 





的 属性 相同 ; 
(4) SELECT 语句 参数 是 一 个 完整 的 查询 语句 ， 表 示 从 某 个 表 中 查 出 某 些 满足 条 件 的 记录 ， 将 这 
些 记 录 导 入 视图 中 ; 


(5) WITH CHECK OPTION 是 可 选 参 数 ， 表 示 更 新 视图 时 要 保证 在 该 视图 的 权限 范围 之 内 。 

例 12.2 在 数据 表 tb_book 中 创建 viewl 视图 ， 视 图 命名 为 book_view1， 并 设置 视图 属性 分 别 为 
a_sort、a_talk、a_books， 代 码 如 下 。 (实例 位 置 光盘 \TMNsM\2\12.2) 

CREATE VIEW 

book_view1(a_sort,a_talk,a_books) 

AS SELECT sort,talk,books 

FROM tb_book:; 


执行 结果 如 图 12.2 所 示 。 





图 12.2 创建 视图 book_view1 
如 果 要 在 tb_book 表 和 tb_user 表 上 创建 名 为 book_view1 的 视图 ， 执 行 代码 如 下 。 


CREATE ALGORITHM=MERGE VIEW 
book_view1(a_sort,a_talk,a_books,a_name) 

AS SELECT sort,talk,books,tb_user.name 

FROM tb_book,tb_name WHERE tb_book.id=tb_name.id 
WITH LOCAL CHECK OPTION; 


例 12.3 在 实际 项 目 开 发 过 程 中 的 数据 表 中 可 能 有 很 多 的 字段 ， 但 某 个 模块 可 能 只 需要 其 中 的 几 
个 。 为 了 提高 查询 速度 和 简化 操作 ， 可 以 将 该 模块 需要 的 字段 单独 提取 出 来 放 在 某 个 视图 中 ， 例 如 ， 
本 实例 涉及 学 生 表 和 成 绩 表 ， 在 建立 的 视图 中 只 含有 与 学 生成 绩 有 关 的 字段 ， 如 图 12.3 所 示 。 《实例 
位 置 ， 光盘 \TMNsIN12\12.3) 
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图 12.3 创建 视图 


运行 本 实例 ， 通 过 MySQL 视图 查询 学 生成 绩 信息 ， 运 
容 即 为 视图 中 所 有 字段 的 内 容 。 





了 结果 如 图 12.4 所 示 ， 查 询 结果 显示 的 内 














0312315 上 50 

0312316 全 星星 6 85 76 
0312317 黄 小 全 5 9 5 
0312318 李 小 林 四 56 78 


























图 12.4 学 生成 绩 列表 
(1) 创建 视图 scoreinfo， 通 过 该 视图 显示 学 生成 绩 信息 ， 该 视图 的 创建 代码 如 下 所 示 。 
create view scoreinfo as select sno,sname,yw,wy,Sx from tb_student,tb_score where tb_student.id=tb_score.sid 


(2) 建立 数据 库 连 接 文件 conn.php 实现 与 MySQL 数据 库 的 连接 , 并 设置 数据 库 字符 集 为 UTF-8， 
代码 如 下 。 





<?php 

$conn=new mysqli("localhost","root","root","db_database12"); /| 连接 数据 库 
$conn->query("set names utf8"); // 设 置 编码 格式 
> 


(3) 查询 视图 scoreinfo 中 的 内 容 ， 并 显示 查询 结果 ， 代 码 如 下 。 


<?php 
include_once("conn.php"); /包含 conn.php 
$sql=$conn->query("select * from scoreinfo"); /执行 查询 
$info=$sql->fetch_array(MYSQLIL ASSOC): // 获 得 查询 结果 集 
if($info==NULLX /判断 是 否 查 询 到 成 绩 信息 
echo " 暂 无 学 生 信息 "; 
jelse{ 
dof // 通 过 循环 打印 学 生成 绩 信息 
Ke 
<tr> 


<td height="20" bgcolor="#FFFFFF"><div align="center"><?php echo $info[sno];?></div></td> 
<td bgcolor="#FFFFFF"><div align="center"><?php echo Sinfo[sname];?></div></td> 
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<td bgcolor="#FFFFFF"><div align="center"><?php echo S$info[yw];?></div></td> 
<td bgcolor="#FFFFFF"><div align="center"><?php echo S$info[wy];?></div></td> 
<td bgcolor="#FFFFFF"><div align="center"><?php echo S$info[sx];?></div></td> 
</tr> 
<?php 
}while($info=$sql->fetch_array(MYSQLI_ASSOC)); 


} 
?> 


12.2.3 ”创建 视图 的 注意 事项 


创建 视图 时 需要 注意 以 下 几 点 。 

(1) 运行 创建 视图 的 语句 需要 用 户 具有 创建 视图 (create view) 的 权限 ， 若 加 了 [or replace] 时 ， 还 
需要 用 户 具有 删除 视图 (drop view) 的 权限 。 

(2) SELECT 语句 不 能 包含 FROM 子 句 中 的 子 查 询 。 

(3) SELECT 语句 不 能 引用 系统 或 用 户 变 量 。 

(4) SELECT 语句 不 能 引用 预 处 理 语句 参数 。 

(5) 在 存储 子 程序 内 ， 定 义 不 能 引用 子 程序 参数 或 局 部 变量 。 

(6) 在 定义 中 引用 的 表 或 视图 必须 存在 。 但 是 ， 创 建 了 视图 后 ， 能 够 舍弃 定义 引用 的 表 或 视图 。 
要 想 检查 视图 定义 是 否 存 在 这 类 问题 ， 可 使 用 CHECK TABLE 语句 。 

(7) 在 定义 中 不 能 引用 temporary 表 ， 不 能 创建 temporary 视图 。 

(8) 在 视图 定义 中 命名 的 表 必 须 已 存在 。 

(9) 不 能 将 触发 程序 与 视图 关联 在 一 起 。 

(10) 在 视图 定义 中 允许 使 用 order by, 但 是 ， 如 果 从 特定 视图 进行 了 选择 ， 而 该 视图 使 用 了 有 具有 
自己 order by 的 语句 ， 它 将 被 忽略 。 








12.3 视图 操作 


12.3.1 查看 视图 


查看 视图 是 指 查看 数据 库 中 已 存在 的 视图 。 查 看 视图 必须 要 有 SHOW VIEW 的 权限 。 查 看 视图 的 
方法 主要 包括 DESCRIBE 语句 、SHOW TABLE STATUS 语句 、SHOW CREATE VIEW 语句 等 。 本 节 
将 主要 介绍 这 几 种 查看 视图 的 方法 。 


1. DESCRIBE 语句 


DESCRIBE 可 以 缩写 成 DESC， 其 语法 格式 如 下 。 
DESCRIBE 视图 名 ; 
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下 面 使 用 DESC 语句 查询 book_view1l 视图 中 的 结构 ， 结 果 如 图 12.5 所 示 。 





图 12.5 使 用 DESC 语句 查询 book viewl 视图 中 的 结构 


结果 中 显示 了 字段 的 名 称 (Field) 、 数 据 类 型 (Type) 、 是 否 为 空 (Null) 、 是 否 为 主 外 键 (Key) 、 
默认 值 (Default) 和 额外 信息 (Extra) 。 
/说明 
如 果 只 需 了 解 视图 中 的 各 个 字段 的 简单 信息 ， 可 以 使 用 DESCRIBE 语句 。DESCRIBE 语句 查 
看 视图 的 方式 与 查看 普通 表 的 方式 是 相同 的 , 结果 显示 的 方式 也 相同 ,通常 情况 下 , 都 是 使 用 DESC 
代替 DESCRIBE。 


2. SHOW TABLE STATUS 语句 

在 MySQL 中 ， 可 以 使 用 SHOW TABLE STATUS 语句 查看 视图 的 信息 ， 其 语法 格式 如 下 。 

SHOW TABLE STATUS LIKE ' 视 图 名 '; 

(1) “LIKE” 表 示 后 面 匹 配 的 是 字符 串 ; 

(2) “视图 名 ”参数 指 要 查看 的 视图 名 称 ， 需 要 用 单 引 号 定义 。 

例 12.4 下 面 使 用 SHOW TABLE STATUS 语句 查看 视图 book viewl 中 的 信息 ， 代 码 如 下 。 ( 实 
例 位 置 ， 光盘 \TMNsIN12\12.4) 


SHOW TABLE STATUS LIKE 'book_view1'; 
执行 结果 如 图 12.6 所 示 。 





WY TABLE STATUS LIKE 
1. 








图 12.6 使 用 SHOW TABLE STATUS 语句 查看 视图 book_viewl 中 的 信息 
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从 执行 结果 可 以 看 出 ， 存 储 引擎 、 数 据 长 息 都 显示 为 NULL， 则 说 明 视 图 为 虚拟 表 ， 与 普通 
数据 表 是 有 区 别 的 。 下面 使 用 SHOW TABLE STATUS 语句 来 查看 tb_book 表 的 信息 , 执行 结果 如 图 12.7 
所 示 。 








ysql> SHOW TABL TUS LIKE ’th_book 
和 
th_hook 
MyISRH 
19 


Row_forma 
R 
fvg_row_length 


utf8_general_ci 
NULL 
e_options 
Commen 
set 《9.98 sec) 





图 12.7 使 用 SHOW TABLE STATUS 语句 来 查看 tb_book 表 的 信息 

从 上 面 的 结果 中 可 以 看 出 ， 数 据 表 的 信息 都 已 经 显示 出 来 了 ， 这 就 是 视图 和 普通 数据 表 的 区 别 。 

3. SHOW CREATE VIEW 语句 

在 MySQL 中 ，SHOW CREATE VIEW 语句 可 以 查看 视图 的 详细 定义 ， 其 语法 格式 如 下 。 

SHOW CREATE VIEW 视图 名 

例 12.5 下 面 使 用 SHOW CREATE VIEW 语句 查看 视图 book_viewl 的 信息 ， 代 码 如 下 。 《实例 
位 置 ， 光盘 \TMNsIN12\12.5) 

SHOW CREATE VIEW book_view1; 

代码 执行 结果 如 图 12.8 所 示 。 


Inysql> SHOW CREATE VIEW hook_viewl\G; 
守 尖 关 迁 尖 源源 源源 江浙 六 尖 尖 泊 尖 关 尖 浙江 源源 关 浙江 二 。 有 OW 和 济源 关 关 浙江 六 关 拓 泊 源源 尖 关 泊 并 关 和 江浙 


book_viewl 
CREATE ALGORITHM=UNDEFINED DEFINER="“root“@“localhost" SQL SECURITY 


“hook viewl* AS select ‘th_ book'.‘sort' fAS ‘a_sort',.‘th_ hook'.'talk 
”RS “a hooks. from “th_hook. 


1 row in set (0.08 sec) 





图 12.8 使 用 SHOW CREATE VIEW 语句 查看 视图 book_view1 的 信息 


通过 SHOW CREATE VIEW 语句 ， 可 以 查看 视图 的 所 有 信息 。 
12.3.2 ”修改 视图 


: 段 发 生 改 变 时 ， 可 以 通过 修改 








修改 视图 是 指 修改 数据 库 中 已 存在 的 表 的 定义 。 当 基本 表 的 某 


208 


第 12 章 视 图 


视图 来 保持 视图 和 基本 表 之 间 一 致 。MySQL 中 通过 CREATE OR REPLACE VIEW 语句 和 ALTER 语 
句 来 修改 视图 。 下 面 介绍 这 两 种 修改 视图 的 方法 。 


1. CREATE OR REPLACE VIEW 


在 MySQL 中 ，CREATE OR REPLACE VIEW 语句 可 以 用 来 修改 视图 。 该 语句 的 使 用 非常 灵活 。 
已 经 存在 的 情况 下 ， 对 视图 进行 修改 ， 视 图 不 存在 时 ， 可 以 创建 视图 。CREATE OR REPLACE 
W 语句 的 语法 如 下 。 


CREATE OR REPLACE [ALGORITHM={UNDEFINED | MERGE | TEMPTABLE)}] 
VIEW 视图 [( 属 性 清单 )] 

AS SELECT 语句 

[WITH [CASCADED | LOCAL] CHECK OPTION]:; 





例 12.6 ”下面 使 用 CREATE OR REPLACE VIEW 语句 将 视图 book_view1 的 字段 修改 为 a_sort 和 
a_book， 执 行 结果 如 图 12.9 所 示 。 (实例 位 置 : 光盘 \TMsIM12\12.6) 





图 12.9 使 用 CREATE OR REPLACE VIEW 语句 修改 视图 
使 用 DESC 语句 查询 book_viewl 视图 ， 结 果 如 图 12.10 所 示 。 





“8.60 se 


图 12.10 使 用 DESC 语句 查询 book_view1 
从 上 面 的 结果 中 可 以 看 出 ， 修 改 后 的 book_viewl 中 只 有 两 个 字段 。 
2. ALTER 
ALTER VIEW 语句 改变 了 视图 的 定义 , 包括 被 索引 视图 , 但 不 影响 所 依赖 的 存储 过 程 或 触发 器 。 
该 语句 与 CREATE VIEW 语句 有 着 同样 的 限制 ， 如 果 删 除 并 重建 了 一 个 视图 ， 就 必须 重新 为 它 分 配 


权限 。 
ALTER VIEW 语句 的 语法 如 下 。 


alter view [algorithm={merge | temptable | undefined} ]view view_name [(column_list)] as select_statement[with 
[cascaded | local] check option] 

(1) algorithm: 该 参数 已 经 在 创建 视图 中 做 了 介绍 ， 这 里 不 再 装 述 。 

(2) view_name: 视图 的 名 称 。 

(3) select_statement: SQL 语句 用 于 限定 视图 。 
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和 as 注 意 
在 创建 视图 时 , 在 使 用 了 WITH CHECK OPTION, WITH ENCRYPTION,WITH SCHEMABING 
或 VIEW_ METADATA 选项 时 ， 如 果 想 保留 这 些 选 项 提供 的 功能 ， 必 须 在 ALTER VIEW 语句 中 将 
它们 包括 进去 。 


例 12.7 ”下面 将 对 book viewl 视图 进行 修改 ,将 原 有 的 a_sort 和 a_book 两 个 属性 更 改 为 a_sort 
个 属性 。 在 更 改 前 ， 先 来 查看 一 下 book viewl 视图 此 时 包含 的 属性 ， 结 果 如 图 12.11 所 示 。 “实例 位 
置 : 光盘 \TMsI\12\12.7) 





图 12.11 查看 book view1l 视图 的 属性 
从 结果 中 可 以 看 出 ,此 时 的 book viewl 视图 中 包含 两 个 属性 , 下 面 对 视 图 进行 修改 ,结果 如 图 12.12 
所 示 。 


> ALTER VIEYW 
AS SELECT 


> FROM th 


> WITH CHECK OPTION; 
Query OK. 8 rows affected 8.00 sec》 


图 12.12 修改 视图 属性 
结果 显示 修改 成 功 ， 下 面 再 来 查看 一 下 修改 后 的 视图 属性 ， 结 果 如 图 12.13 所 示 。 


iewl; 








set 《0.60 sec) 


图 12.13 查看 修改 后 的 视图 属性 


结果 显示 ， 此 时 视图 中 只 包含 一 个 a_sort 属性 。 





12.3.3 ”更 新 视图 


对 视图 的 更 新 其 实 就 是 对 表 的 更 新 ， 更 新 视图 是 指 通过 视图 来 插入 (INSERT) 、 更 新 (UPDATE) 
和 删除 (DELETE) 表 中 的 数据 。 因 为 视图 是 一 个 虚拟 表 ， 其 中 没有 数据 。 通 过 视图 更 新 时 ， 都 是 转 
换 到 基本 表 来 更 新 。 更 新 视图 时 ， 只 能 更 新 权限 范围 内 的 数据 ， 超 出 了 范围 ， 就 不 能 更 新 。 本 节 讲 解 
更 新 视图 的 方法 和 更 新 视图 的 限制 。 
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1. 


例 12.8 下 面 对 book view2 视图 中 


所 示 。 


更 新 视图 
ph 的 数据 进行 更 新 , 先 来 查看 book_view2 视图 中 的 数据 , 如 图 12.14 





(实例 位 置 ， 光 盘 \TMsIN12\12.8) 





图 12.14 查看 book view2 视图 中 的 数据 


下 面 更 新 视图 中 的 第 27 条 记录 ，a_sort 的 值 为 “模块 类 ”，a_book 的 值 为 “PHP 典型 模块 ”， 更 


新 语句 结果 如 图 12.15 所 示 。 


所 示 。 


《B-B1 s 
: 工 





图 12.15 更 新 视图 中 的 数据 


结果 显示 更 新 成 功 ， 下 面 再 来 查看 一 下 book_view2 视图 中 的 数据 是 否 有 变化 ， 结 果 如 图 12.16 


并 可 





图 12.16 查看 更 新 后 视图 中 的 数据 
结果 如 图 12.17 所 示 。 


此 不 


下 面 再 来 查看 一 下 tb_book 表 中 的 数据 是 否 有 变化 ， 





图 12.17 查看 tb_book 表 中 的 数据 


从 上 面 的 结果 可 以 看 出 ， 对 视图 的 更 新 其 实 就 是 对 基本 表 的 更 新 。 


2. 更 新 视图 的 限制 
不 是 所 有 的 视图 都 可 以 更 新 ， 以 下 几 种 情况 是 不 能 更 新 视图 的 。 





并 
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(1) 视图 中 包含 COUNTO、SUM0O、MAX0 和 MINO 等 函数 。 例 如 : 


CREATE VIEW book_view1(a_sort,a_book) 
AS SELECT sort,books, COUNT(name) FROM tb_book: 


(2) 视图 中 包含 UNION、UNION ALL、DISTINCT、GROUP BY 和 HAVIG 等 关键 字 。 例 如 : 
CREATE VIEW book_view1(a_sort,a_book) 
AS SELECT sort,books, FROM tb_book GROUP BY id; 
(3) 常量 视图 。 例 如 : 
CREATE VIEW book_view1 
AS SELECT 'Aric as a_book; 
(4) 视图 中 的 SELECT 中 包含 子 查询 。 例 如 : 
CREATE VIEW book_view1(a_sort) 
AS SELECT (SELECT name FROM tb_book); 
(5) 由 不 可 更 新 的 视图 导出 的 视图 。 例 如 : 
CREATE VIEW book_view1 
AS SELECT * FROM book_view2; 
(6) 创建 视图 时 ，ALGORITHM 为 TEMPTABLE 类 型 。 例 如 : 


CREATE ALGORITHM=TEMPTABLE 
VIEW book_view1 
AS SELECT * FROM tb_book; 

(7) 视图 对 应 的 表 上 存在 没有 默认 值 的 列 , 而 且 该 列 没有 包含 在 视图 里 。 例如， 表 中 包含 的 name 
字段 没有 默认 值 ， 但 是 视图 中 不 包括 该 字段 ， 那 么 这 个 视图 是 不 能 更 新 的 。 因 为 ， 在 更 新 视图 时 ， 这 
个 没有 默认 值 的 记录 将 没有 值 插入 , 也 没有 NULL 值 插入 。 数 据 库 系统 是 不 会 允许 这 样 的 情况 出 现 的 ， 
其 会 阻止 这 个 视图 更 新 。 

上 面 的 几 种 情况 其 实 就 是 一 种 情况 ， 即 视图 的 数据 和 基本 表 的 数据 不 一 样 了 。 


< 人 0 注意 

视图 中 虽然 可 以 更 新 数据 , 但 是 有 很 多 的 限制 。 一 般 情 况 下 ,最 好 将 视图 作为 查询 数据 的 虚拟 
表 ， 而 不 要 通过 视图 更 新 数据 。 因 为 ,使 用 视图 更 新 数据 时 ， 如 果 没 有 全 面 考虑 在 视图 中 更 新 数据 
的 限制 ， 可 能 会 造成 数据 更 新 失败 。 


12.3.4 删除 视图 


删除 视图 是 指 删除 数据 库 中 已 存在 的 视图 。 删 除 视图 时 ， 只 能 删除 视图 的 定义 ， 不 会 删除 数据 。 
MySQL 中 ,使 用 DROP VIEW 语句 来 删除 视图 。 但 是 ,用 户 必 须 拥 有 DROP 权限 。 本 节 将 介绍 删除 视 
图 的 方法 。 
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DROP VIEW 语句 的 语法 如 下 。 

DROP VIEW IF EXISTS < 视图 名 > [RESTRICT | CASCADE] 

(1) 正 EXISTS 参数 指 判断 视图 是 否 存在 ， 如 果 存 在 则 执行 ， 不 存在 则 不 执行 。 

(2) “视图 名 ”列表 参数 表示 要 删除 的 视图 的 名 称 和 列表 ， 各 个 视图 名 称 之 间 用 逗号 隔 开 。 

该 语句 从 数据 字典 中 删除 指定 的 视图 定义 ;如果 该 视图 导出 了 其 他 视图 ， 则 使 用 CASCADE 级 联 
删除 ， 或 者 先 显 式 删除 导出 的 视图 ， 再 删除 该 视图 ;删除 基 表 时 ， 由 该 基 表 导出 的 所 有 视图 定义 都 必 
须 显 式 删除 。 


例 12.9 下 面 删除 前 面 实例 中 一 直 使 用 的 book viewl 视图 ， 执 行 语句 如 下 。 《实例 位 置 ， 光 盘 \ 
TM™MNsN\12\12.9) 

















DROP VIEW IF EXISTS book_view1; 
执行 结果 如 图 12.18 所 示 。 





12.18 ”删除 视图 


执行 结果 显示 删除 成 功 。 下 面 验证 一 下 视图 是 否 真正 被 删除 ,执行 SHOW CREATE VIEW 语句 查 
看 ， 执 行 结果 如 图 12.19 所 示 。 
Im 1> SHOW CREATE UIEW b 

EF 146 《42882): Tal 








图 12.19 查看 视图 是 否 删 除 成 功 


结果 显示 ， 视 图 book_view1l 已 经 不 存在 了 ， 说 明 DROP VIEW 语句 删除 视图 成 功 。 


12.4 小 结 


本 章 对 MySQL 数据 库 的 视图 的 含义 和 作用 进行 了 详细 讲解 ， 并 且 讲 解 了 创建 视图 、 修 改 视图 和 
删除 视图 的 方法 。 创 建 视 图 和 修改 视图 是 本 章 的 重点 内 容 ， 并 且 需 要 在 计算 机 上 实际 操作 。 读 者 在 创 
建 视图 和 修改 视图 后 ， 一 定 要 查看 视图 的 结构 ， 以 确保 创建 和 修改 的 操作 正确 。 更 新 视图 是 本 章 的 一 
个 难点 ， 因 为 实际 中 存在 一 些 造成 视图 不 能 更 新 的 因素 ， 希 望 读者 在 练习 中 认真 分 析 。 











12.5 ”实践 与 练习 


1. 在 department 表 上 创建 一 个 简单 的 视图 ， 视 图 名 称 为 department_view1。〔 答 案 位 置 ， 光盘 
\TMNsN\12\12.10) 


2. 使 用 DESCRIBE 语句 查询 视图 的 结构 。 (答案 位 置 ， 光盘 \ITMNsN2\12.11) 
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第 13 章 数据 完整 性 约束 

第 14 章 ， 存储 过 程 与 存储 田 数 

第 15 章 触发 路 

第 16 章 事务 的 应 用 

第 17 章 事件 

第 18 章 备份 与 恢复 

第 19 章 MySQL 性 能 优化 

第 20 章 ”权限 管理 及 安全 控制 

第 21 章 PHP 管理 MySQL 数据 库 中 的 数据 


豆 于 于 竺 于 于 革 于 至 


本 篇 介绍 数据 完整 性 约束 、 存储 过 程 与 存储 函数 、 触 发 器 、 事 务 的 应 用 、 事 件 、 
备份 与 恢复 、MySQL 性 能 优化 、 权 限 管 理 及 安全 控制 、PHP 管理 MySQL 数据 库 
中 的 数据 等 。 学 习 完 这 一 部 分 ， 能 够 境 担 如 何 进行 数据 的 导入 与 导出 操作 ， 以 及 使 
用 存储 过 程 、 触 发 器 、 事 务 、 事 件 等 。 通 过 这 些 内 容 不 仅 可 以 优化 查询 ， 还 可 以 提 
高 数据 访问 速度 ; 更 好 地 维护 MySQL 的 权限 及 其 安全 。 另 外 ， 还 介绍 了 应 用 PHP 
管理 My5QL 数据 库 中 的 数据 ， 对 于 想 要 使 用 PHP 开发 的 读者 非常 实用 。 


第 《了 


数据 完整 性 约束 


数据 完整 性 是 指数 据 的 正确 性 和 相 窒 性， 是 为 了 防止 数据 库 中 存在 不 符合 语义 
的 数据 ， 即 防止 数据 库 中 存在 不 正确 的 数据 。 在 MySQL 中 提供 了 多 种 完整 性 约束 ， 
它们 作为 数据 库 关 系 模式 定义 的 一 部 分 ,可 以 通过 CREATE TABLE 或 ALTER TABLE 
语句 来 定义 。 一 旦 定义 了 完整 性 约束 ，IMy5QL 服务 器 会 随时 检测 处 于 更 新 状态 的 
数据 库 内 容 是 否 符合 相关 的 完整 性 约束 ， 从 而 保证 数据 的 一 致 性 与 正确 性 。 这 样 
既 能 有 效 地 防止 对 数据 库 的 意外 破坏 ,又 能 提高 完整 性 检测 的 效率 ， 还 能 减轻 数据 
库 编程 人 员 的 工作 人 负担。 本章 中 将 对 数据 完整 性 约束 进行 详细 介绍 。 

通过 阅读 本 章 ， 读者 可 以 : 


MW 掌握 定义 完整 性 约束 的 方法 


M 掌握 命名 完整 性 约束 的 方法 
M 掌握 更 新 完整 性 约束 的 方法 


第 13 章 数据 完整 性 约束 


13.1 定义 完整 性 约束 


关系 模型 的 完整 性 规则 是 对 关系 的 某 种 约束 条 件 。 在 关系 模型 中 ， 提 供 了 实体 完整 性 、 参 照 完 
整 性 和 用 户 定义 完整 性 3 项 规则 。 下 面 将 分 别 介绍 MySQL 中 对 数据 库 完整 性 3 项 规则 的 设置 和 实现 
方式 。 


13.1.1 实体 完整 性 


实体 (Entity) 是 一 个 数据 对 象 ， 是 指 客观 存在 并 可 以 相互 区 分 的 事物 ， 如 一 个 教师 、 一 个 学 生 或 
一 个 雇员 等 。 一 个 实体 在 数据 库 中 表现 为 表 中 的 一 条 记录 。 通 常情 况 下 ， 它 必须 遵守 实体 完整 性 规则 。 

实体 完整 性 规则 (Entity Integrity Rule) 是 指 关系 的 主 属性 ， 即 主 码 〈 主 键 ) 的 组 成 不 能 为 室 ， 也 
就 是 关系 的 主 属性 不 能 是 空 值 (NULL) 。 关 系 对 应 于 现实 世界 中 的 实体 集 ， 而 现实 世界 中 的 实体 是 可 
区 分 的 ， 即 说 明 每 个 实例 具有 唯一 性 标识 。 在 关系 模型 中 ， 是 使 用 主 码 〈 主 键 ) 作为 唯一 性 标识 的 
若 假设 主 码 (主键 ) 取 空 值 ， 则 说 明 这 个 实体 不 可 标识 ， 即 不 可 区 分 ， 这 个 假设 显然 不 正确 ， 与 现实 
世界 应 用 环境 相 矛 盾 ， 因 此 不 能 存在 这 样 的 无 标识 实体 ， 从 而 在 关系 模型 中 引入 实体 完整 性 约束 。 例 
如 ， 学 生 关系 (学 号 、 姓 名 、 性 别 )》 中 ，“ 学 号 ”为 主 码 (主键 )， 则 “学 号 ”这 个 属性 不 能 为 空 值 ， 
否则 就 违反 了 实体 完整 性 规则 。 

在 MySQL 中， 实体 完整 性 是 通过 主键 约束 和 候选 键 约束 来 实现 的 。 

1. 主键 约束 

主键 可 以 是 表 中 的 某 一 列 ， 也 可 以 是 表 中 多 个 列 所 构成 的 一 个 组 合 ， 其中， 由 多 个 列 组 合 而 成 的 
主键 也 称 为 复合 主键 。 在 MySQL 中 ， 主 键 列 必须 遵守 以 下 规则 。 

(1) 每 一 个 表 只 能 定义 一 个 主键 。 

(2) 唯一 性 原则 。 主 键 的 值 ， 也 称 键 值 ， 必 须 能 够 唯一 标识 表 中 的 每 一 行 记 录 ， 且 不 能 为 NULL。 
也 就 是 说 一 张 表 中 两 个 不 同 的 行 在 主键 上 不 能 具有 相同 的 值 。 

(3) 最 小 化 规则 。 复 合 主键 不 能 包含 不 必要 的 多 余 列 。 也 就 是 说 ， 当 从 一 个 复合 主键 中 删除 一 列 
后 ， 如 果 剩 下 的 列 构成 的 主键 仍 能 满足 唯一 性 原则 ， 那 么 这 个 复合 主键 是 不 正确 的 。 

(4) 一 个 列 名 在 复合 主键 的 列表 中 只 能 出 现 一 次 。 

在 MySQL 中 ， 可 以 在 CREATE TABLE 或 者 ALTER TABLE 语句 中 ， 使 用 PRIMARY KEY 子 句 
来 创建 主键 约束 ， 其 实现 方式 有 以 下 两 种 。 

1) 作为 列 的 完整 性 约束 

在 表 的 某 个 列 的 属性 定义 时 ， 加 上 PRIMARY KEY 关键 字 实 现 。 

例 13.1 在 创建 用 户 信息 表 tb_user 时 ， 将 id 字段 设置 为 主键 ， 代 码 如 下 。 (实例 位 置 ; 光盘 
VIMsN13\13.1) 








create table tb_user( 
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id int auto_increment primary key, 
user varchar(30) not null, 
password varchar(30) not null, 
createtime datetime); 


运行 上 述 代 码 ， 其 结果 如 图 13.1 所 示 。 

2) 作为 表 的 完整 性 约束 

在 表 的 所 有 列 的 属性 定义 后 ， 加 上 PRIMARY KEY(index_col name,…) 子 句 实 现 。 

例 13.2 创建 学 生 信息 表 tb_student 时 ， 将 学 号 〈id) 和 所 在 班级 号 〈classid) 字段 设置 为 主键 ， 
代码 如 下 。《〈 实 例 位 置 ; 光盘 \TMNsIN13\13.2) 


create table tb_student ( 
id int auto_increment, 
name varchar(30) not null, 
sex varchar(2), 

classid int not null, 
birthday date, 

PRIMARY KEY (id,classid) 
); 


运行 上 述 代 码 ， 其 结果 如 图 13.2 所 示 。 


t prinary key, 
null, 


not null, 


Inysql> - 





图 13.1 将 这 字段 设置 为 主键 图 13.2 将 id 字 段 和 classid 字段 设置 为 主键 


C4% 明 

如 果 主 键 仅 由 表 中 的 某 一 列 所 构成 ,那么 以 上 两 种 方法 均 可 以 定义 主键 约束 ; 如 果 主 键 由 表 中 
多 个 列 所 构成 ， 那么 只 能 用 第 二 种 方法 定义 主键 约束 。 另 外 ， 定 义 主键 约束 后 ，MySQL 会 自动 为 
主键 创建 一 个 唯一 索引 ， 默 认 名 为 PRIMARY， 也 可 以 修改 为 其 他 名 称 。 


2. 候选 键 约束 


如 果 一 个 属性 集 能 唯一 标识 元 组 ， 且 又 不 含有 多 余 的 属性 ， 那 么 这 个 属性 集 称 为 关系 的 候选 键 。 
例如 ， 在 包含 学 号 、 姓 名 、 性 别 、 年 龄 、 院 系 、 班 级 等 列 的 “学 生 信息 表 ” 中 ，“ 学 号 ”能 够 标识 一 
名 学 生 ， 因 此 ， 它 可 以 作为 候选 键 ， 而 如 果 规 定 ， 不 允许 有 同名 的 学 生 ， 那 么 姓名 也 可 以 作为 候选 键 。 

候选 键 可 以 是 表 中 的 某 一 列 ， 也 可 以 是 表 中 多 个 列 所 构成 的 一 个 组 合 。 任 何 时 候 ， 候 选 键 的 值 必 
须 是 唯一 的 ， 且 不 能 为 空 (NULL) 。 候 选 键 可 以 在 CREATE TABLE 或 者 ALTER TABLE 语句 中 使 用 
关键 字 UNIQUE 来 定义 ， 其 实现 方法 与 主键 约束 类 似 ， 也 是 可 作为 列 的 完整 性 约束 或 者 表 的 完整 性 约 
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束 两 种 方式 。 

在 MySQL 中 ， 候 选 键 与 主键 之 间 存 在 以 下 两 点 区 别 。 

(1) 一 个 表 只 能 创建 一 个 主键 ， 但 可 以 定义 若干 个 候选 键 。 

(2) 定义 主键 约束 时 ， 系 统 会 自动 创建 PRIMARY KEY 索引 ， 而 定义 候选 键 约束 时 ， 系 统 会 自动 
创建 UNIQUE 索引 。 

例 13.3 在 创建 用 户 信息 表 tb_userl 时 ,将 id 字段 和 user 字段 设置 为 候选 键 ， 代 码 如 下 。 (实例 
位 置 ， 光盘 \TMsIN13\13.3) 





create table tb_user1( 

id int auto_increment UNIQUE， 

user varchar(30) not null UNIQUE， 

password varchar(30) not null, 

createtime TIMESTAMP default CURRENT_TIMESTAMP); 


运行 上 述 代码 ， 其 结果 如 图 13.3 所 示 。 





图 13.3 将 这 字段 和 user 字段 设置 为 候选 键 
13.1.2 ”参照 完整 性 


现实 世界 中 的 实体 之 间 往 往 存在 着 某 种 联系 ， 在 关系 模型 中 ， 实 体 及 实体 间 的 联系 都 是 用 关系 来 
描述 的 ， 那 么 自然 就 存在 着 关 间 的 引用 。 例 如 ， 学 生 实体 和 班级 实体 可 以 分 别 用 下 面 的 关系 
表示 ， 其 中 ， 主 码 〈 主 键 ) 用 下 划 线 标识 。 

学 生 号 ， 姓 名 ， 性 别 ， 生 日 ， 班 级 编号 ， 备 注 ) 

班级 〈 班 级 编号 ， 班 级 名 称 ， 备 注 ) 

在 这 两 个 关系 之 间 存 在 着 属性 的 引用 ， 即 “学 生 ” 关 系 引 用 了 “班级 ”关系 中 的 主 码 (主键 ) “ 班 
级 编号 ”。 在 两 个 实体 间 ，“ 班 级 编号 ”是 “班级 ”关系 的 主 码 ( 主 键 ) ， 也 是 “学 生 ” 关 系 的 外 部 
码 ( 外 键 ) 。 显 然 ， “学生 ”关系 中 的 “班级 编号 ”的 值 必须 是 确实 存在 的 班级 的 “班级 编号 ”， 即 
“班级 ”关系 中 的 该 班级 的 记录 。 也 就 是 说 ，“ 学 生 ” 关 系 中 某 个 属性 的 取 值 需要 参照 “班级 ”关系 
的 属性 和 值 。 

参照 完整 性 规则 (Referential Integrity Rule) 就 是 定义 外 码 〈 外 键 》 和 主 码 (主键) 之 间 的 引用 规 
则 ， 它 是 对 关系 间 引 用 数据 的 一 种 限制 。 

参照 完整 性 的 定义 为 : 若 属性 〈 或 属性 组 ) F 是 基本 关系 及 的 外 码 ， 它 与 基本 关系 $S 的 主 码 K 相对 
应 ， 则 对 于 R 中 每 个 元 组 在 F 上 的 值 只 允许 两 种 可 能 ， 即 要 么 取 空 值 GF 的 每 个 属性 值 均 为 空 值 ) ， 要 
么 等 于 S 中 某 个 元 组 的 主 码 值 。 其 中 ， 关 系 及 与 S 可 以 是 不 同 的 关系 ,也 可 以 是 同一 关系 , 而 F 与 KK 是 
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定义 在 同一 个 域 中 。 例 如 ， 在 “学 生 ” 关 系 中 每 个 学 生 的 “班级 编号 ”一 项 ， 要 么 取 空 值 ， 表 示 该 学 生 
还 没有 分 配 班级 ， 要 么 取 值 必须 与 “班级 ”关系 中 的 某 个 元 组 的 “班级 编号 ”相同 ， 表 示 这 个 学 生 分 配 
到 某 个 班级 学 习 。 这 就 是 参照 完整 性 。 如 果 “ 学 生 ” 关 系 中 ， 某 个 学 生 的 “班级 编号 ” 取 值 不 能 与 “ 班 
级 ”关系 中 任何 一 个 元 组 的 “班级 编号 ” 值 相同 ， 表 示 这 个 学 生 被 分 配 到 不 属于 所 在 学 校 的 班级 学 习 ， 
这 与 实际 应 用 环境 不 相符 ， 显 然 是 错误 的 ， 这 就 需要 在 关系 模型 中 定义 参照 完整 性 进行 约束 。 

与 实体 完整 性 一 样 ， 参照 完 整 性 也 是 由 系统 自动 支持 的 ， 即 在 建立 关系 ( 表 ) 时 ， 只 要 定义 了 “ 谁 
是 主 码 ”“ 谁 参照 于 认证 ”， 系 统 将 自动 进行 此 类 完整 性 的 检查 。 在 MySQL 中 ， 参 照 完 整 性 可 以 通 
过 在 创建 表 (CREATE TABLE) 或 者 修改 表 (ALTER TABLE) 时 定义 一 个 外 键 声明 来 实现 。 

MySQL 有 两 种 常用 的 引擎 类 型 (MyISAM 和 InnoDB) ， 目 前 ， 只 有 InnoDB 引擎 类 型 支持 外 键 
约束 。InnoDB 引擎 类 型 中 声明 外 键 的 基本 语法 格式 如 下 。 


[CONSTRAINT [SYMBOL]I 
FOREIGN KEY (index_col_name,…) reference_definition 





reference_definition 主要 用 于 定义 外 键 所 参照 的 表 、 列 、 参 照 动作 的 声明 和 实施 策略 等 4 部 分 内 容 。 
它 的 基本 语法 格式 如 下 。 
REFERENCES tbl_name [(index_col_name,*…*)] 
[MATCH FULL | MATCH PARTIAL | MATCH SIMPLE] 


[ON DELETE reference_option] 
[ON UPDATE reference_option] 


index_col_name 的 语法 格式 如 下 。 

col_name [(length)] [ASC | DESC] 

reference_option 的 语法 格式 如 下 。 

RESTRICT | CASCADE | SET NULL | NO ACTION 

参数 说 明 如 下 。 

(1) index_col name: 用 于 指定 被 设置 为 外 键 的 列 。 

(2) tbl name: 用 于 指定 外 键 所 参照 的 表 名 。 这 个 表 称 为 参照 表 ( 或 父 表 ) ， 而 外 键 所 在 的 表 称 
作 参 照 表 (或 子 表 )。 

(3) col name: 用 于 指定 被 参照 的 列 名 。 外 键 可 以 引用 被 参照 表 中 的 主键 或 候选 键 ， 也 可 以 引用 
被 参照 表 中 某 些 列 的 一 个 组 合 ， 但 这 个 组 合 不 能 是 被 参照 表 中 随机 的 一 组 列 ， 必 须 保存 该 组 合 的 取 值 
在 被 参照 表 中 是 唯一 的 。 外 键 中 的 所 有 列 值 在 被 参照 表 的 列 中 必须 全 部 存在 ， 也 就 是 通过 外 键 来 对 参 
照 表 某 些 列 ( 外 键 ) 的 取 值 进行 限定 与 约束 。 

(4) ON DELETE | ON UPDATE: 指定 参照 动作 相关 的 SQL 语句 。 可 为 每 个 外 键 指定 对 应 于 
DELETE 语句 和 UPDATE 语句 的 参照 动作 。 

(5) reference_ option: 指定 参照 完整 性 约束 的 实现 策略 。 其 中 ， 当 没有 明确 指定 参照 完整 性 的 实 
现 策略 时 ， 两 个 参照 动作 会 默认 使 用 RESTRICT。 具 体 的 策略 可 选 值 如 表 13.1 所 示 。 
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表 13.1 策略 可 选 值 























可 选 值 说 明 
a 限制 策略 : 当 要 删除 或 更 新 被 参照 表 中 被 参照 列 上 ， 并 在 外 键 中 出 现 的 值 时 ， 系 统 拒绝 对 被 参照 
表 的 删除 或 更 新 操作 
CASCADE 级 联 策略 : 从 被 参照 表 中 删除 或 更 新 记录 行 时 ， 自 动 删除 或 更 新 参照 表 匹 配 的 记录 行 
ER 置 空 策略 : 当 从 被 参照 表 中 删除 或 更 新 记录 行 时 , 设置 参照 表 中 与 之 对 应 的 外 键 列 的 值 为 NULL。 
这 个 策略 需要 被 参照 表 中 的 外 键 列 没有 声明 限定 词 NOTNULL 
二 不 采取 实施 策略 : 当 一 个 相关 的 外 键 值 在 被 参照 表 中 时 ， 删 除 或 更 新 被 参照 表 中 键 值 的 动作 不 被 


人 允许。 该 策略 的 动作 语言 与 RESTRICT 相同 








例 13.4 创建 学 生 信 息 表 tb_studentl， 并 为 其 设置 参照 完整 性 约束 《拒绝 删除 或 更 新 被 参照 表 中 


被 参照 列 上 的 外 键 值 ) ， 即 将 classid 字段 设置 为 外 键 ， 代 码 如 下 。 《实例 位 置 : 光盘 \TMNsN13\ 13.4) 


create table tb_student1 ( 
id int auto_increment, 
name varchar(30) not null, 
sex varchar(2), 

classid int not null, 
birthday date, 

remark varchar(100), 
primary key (id), 
FOREIGN KEY (classid) 
REFERENCES tb_class(id) 
ON DELETE RESTRICT 
ON UPDATE RESTRICT 
六 


运行 上 述 代码 ， 其 结果 如 图 13.4 所 示 。 








图 13.4 将 classid 字段 设置 为 外 键 


人 注意 
要 设置 为 主 外 键 关系 的 两 张 数据 表 必 须 具 有 相同 的 存储 引擎 ， 如 都 是 InoDB， 并 且 相 关联 的 
两 个 字段 的 类 型 必须 一 致 。 
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设置 外 键 时， 通常 需 要 遵守 以 下 规则 。 

(1) 被 参照 表 必须 是 已 经 存在 的 ， 或 者 是 当前 正在 创建 的 表 。 如 果 是 当前 正在 创建 的 表 ， 也 就 是 
说 ， 被 参照 表 与 参照 表 是 同一 个 表 ， 这 样 的 表 称 为 自 参 照 表 (Selfreferencing Table) ， 这 种 结构 称 为 
自 参照 完整 性 (Self-referential Integrity) 。 

(2) 必须 为 被 参照 表 定义 主键 。 

(3) 必须 在 被 参照 表 名 后 面 指定 列 名 或 列 名 的 组 合 。 这 个 列 或 列 组 合 必须 是 这 个 被 参照 表 的 主键 
或 候选 键 。 

(4) 外 键 中 列 的 数目 必须 和 被 参照 表 中 的 列 的 数据 相同 。 

(5) 外 键 中 列 的 数据 类 型 必须 和 被 参照 表 的 主键 〈 或 候选 键 ) 中 的 对 应 列 的 数据 类 型 相同 。 

(6) 尽管 主键 是 不 能 够 包含 空 值 的 ， 但 允许 在 外 键 中 出 现 一 个 空 值 。 这 意味 着 ， 只 要 外 键 的 每 个 
非 空 值 出 现在 指定 的 主键 中 ， 这 个 外 键 的 内 容 就 是 正确 的 。 














13.1.3 ”用 户 定义 完整 性 


用 户 定义 完整 性 规则 (User-defined Integrity Rule) 是 针对 某 一 应 用 环境 的 完整 性 约束 条 件 ， 它 反 
映 了 某 一 具体 应 用 所 涉及 的 数据 应 满足 的 要 求 。 关 系 模型 提供 定义 和 检验 这 类 完整 性 规则 的 机 制 ， 其 
目的 古 由 系统 来 统 -处 理 ， 而 不 再 由 应 用 程序 来 完成 这 项 工作 。 在 实际 系统 中 ， 这 类 - 般 
是 在 建立 数据 表 的 同时 进行 定义 ， 应 用 编程 人 员 不 需要 再 做 考虑 ， 如 果 某 些 约束 条 件 没有 建 
-级 ， 则 应 用 编程 人 员 应 在 各 模块 的 具体 编程 中 通过 程序 进行 检查 和 控制 。 
MySQL 支持 非 空 约束 、CHECK 约束 和 触发 器 3 种 用 户 自 定义 完整 性 约束 。 其 中 ， 触 发 器 将 在 第 
15 章 进 行 详细 介绍 。 这 里 主要 介绍 非 空 约束 和 CHECK 约束 。 


1. 非 空 约束 


在 MySQL 中 ， 非 空 约束 可 以 通过 在 CREATE TABLE 或 ALTER TABLE 语句 中 ， 某 个 列 定义 后 
面 加 上 关键 字 NOT NULL 来 定义 ， 用 来 约束 该 列 的 取 值 不 能 为 空 
例 13.5 ”创建 班级 信息 表 tb_class1， 并 为 其 name 字段 添加 非 空 约 束 ， 代 码 如 下 。 (实例 位 置 ; 
光盘 \TMNsIM3\13.5) 
CREATE TABLE tb_class1 ( 
id int(11) NOT NULL AUTO_INCREMENT, 
name varchar(45) NOT NULL., 


remark varchar(100) DEFAULT NULL, 
PRIMARY KEY (id) 

















运行 上 述 代 码 ， 其 结果 如 图 13.5 所 示 。 


| 
NULL AUTO_INCREMENT, 
LL 


Fectea CB.49 





图 13.5 为 name 字段 添加 非 空 约束 
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与 非 空 约束 一 样 , CHECK 约束 也 可 以 通过 在 CREATE TABLE 或 ALTER TABLE 语句 中 , 根据 用 
户 的 实际 完整 性 要 求 来 定义 。 它 可 以 分 别 对 列 或 表 实 施 CHECK 约束 ， 其 中 使 用 的 语法 如 下 。 
CHECK(expr) 


其 中 ，expr 是 一 个 SQL 表达 式 ， 用 于 指定 需要 检查 的 限定 条 件 。 在 更 新 表 数 据 时 ，MySQL 会 检 
查 更 新 后 的 数据 行 是 否 满足 CHECK 约束 中 的 限定 条 件 。 该 限定 条 件 可 以 是 简单 的 表达 式 ， 也 可 以 是 
复杂 的 表达 式 〈 如 子 查询 ) 。 

下 面 将 分 别 介绍 如 何 对 列 和 表 实 施 CHECK 约束 。 

1) 对 列 实施 CHECK 约束 

将 CHECK 子 句 置 于 表 的 某 个 列 的 定义 之 后 就 是 对 列 实施 CHECK 约束 。 下 面 将 通过 一 个 具体 的 
实例 来 说 明 如 何 对 列 实施 CHECK 约束 。 

例 13.6 创建 学 生 信息 表 tb_student2， 限 制 其 age 字段 的 值 只 能 在 7~18 之 间 (不 包括 18) 的 数 ， 
代码 如 下 。 《实例 位 置 : 光盘 \TMNsN13\13.6) 


create table tb_student2 ( 

id int auto_increment， 

name varchar(30) not null, 

sex varchar(2), 

age int not null CHECK(age>6 and age<18), 
remark varchar(100), 

primary key (id) 

jE 














运行 上 述 代 码 ， 其 结果 如 图 13.6 所 示 。 





图 13.6 对 列 实施 CHECK 约束 


WS 说 明 


目前 的 MySQL 版 本 只 是 对 CHECK 约束 进行 了 分 析 处 理 ， 但 会 被 直接 忽略 ， 并 不 会 报错 。 


2) 对 表 实 施 CHECK 约束 

将 CHECK 子 句 置 于 表 中 所 有 列 的 定义 以 及 主键 约束 和 外 键 定义 之 后 就 是 对 表 实 施 CHECK 约束 。 
下 面 将 通过 一 个 具体 的 实例 来 说 明 如 何 对 表 实 施 CHECK 约束 。 

例 13.7 创建 学 生 信 息 表 tb_student3， 限 制 其 classid 字段 的 值 只 能 是 tb_class 表 中 id 字段 的 某 一 
个 这 值 ， 代 码 如 下 。《〈 实 例 位 置 : 光盘 \TMsN13\13.7) 





223 


MySQL 从 入 门 到 精通 


create table tb_student3 ( 

id int auto_increment, 

name varchar(30) not null, 

sex varchar(2), 

classid int not null, 

birthday date, 

remark varchar(100), 

primary key (id), 

CHECK(classid IN (SELECT id FROM tb_class)) 
及 


运行 上 述 代 码 ， 其 结果 如 图 13.7 所 示 。 


SELECT id FROM tb_cla 





图 13.7 ”对 表 实 施 CHECK 约束 
13.2 命名 完整 性 约束 


在 MySQL 中 ， 也 可 以 对 完整 性 约束 进行 添加 、 修 改 和 删除 等 操作 。 其 中 ， 为 了 删除 和 修改 完整 
性 约束 ， 需 要 在 定义 约束 的 同时 对 其 进行 命名 。 命 名 完整 性 约束 的 方式 是 在 各 种 完整 性 约束 的 定义 说 
明之 前 加 上 CONSTRAINT 子 句 实 现 的 。CONSTRAINT 子 句 的 语法 格式 如 下 。 





CONSTRAINT <symbol> 
[PRIMAR KEY 短语 |FOREIGN KEY 短语 |CHECK 短语 ] 


参数 说 明 如 下 。 

(1) symbol: 用 于 指定 约束 名 称 。 这 个 名 字 是 在 完整 性 约束 说 明 的 前 面 被 定义 ， 在 数据 库 里 必须 
是 唯一 的 。 如 果 在 创建 时 没有 指定 约束 的 名 字 ， 则 MySQL 将 自动 创建 一 个 约束 名 字 。 

(2) PRIMAR KEY 短语 : 主键 约束 。 

(3) FOREIGN KEY 短语 : 参照 完整 性 约束 。 

(4) CHECK 短语 : CHECK 约束 。 


wl/ 
说 明 
在 MySQL 中 ,主键 约束 名 称 只 能 是 PRIMARY. 
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例如 ， 对 雇员 表 添加 主键 约束 ， 并 为 其 命名 为 PRIMARY， 可 以 使 用 下 面 的 代码 。 


ALTER TABLE 雇员 表 ADD CONSTRAINT PRIMARY 
PRIMARY KEY (雇员 编号 ) 











例 13.8 修改 例 13.4 的 代码 ， 重 新 创建 学 生 信息 表 tb_student1, 命名 为 tb_studentla,， 并 
完整 性 约束 命名 ， 代 码 如 下 。 《实例 位 置 ， 光盘 \TMNsN13\13.8) 


create table tb_student1a ( 

id int auto_increment PRIMARY KEY, 
name varchar(30) not null, 

sex varchar(2), 

classid int not null, 

birthday date, 

remark varchar(100), 

CONSTRAINT fk_classid FOREIGN KEY (classid) 
REFERENCES tb_class(id) 

ON DELETE RESTRICT 

ON UPDATE RESTRICT 

); 


运行 上 述 代 码 ， 其 结果 如 图 13.8 所 示 。 


e db_database13; 


able thb_studentla 《 
p_increment PRIMARY KEY。 
38> not null, 


TRAINT fk_cla FOREIGN KEY 《classid> 
RE th_cla ) 

DELETE 

UPDATE 了 


0K。B rows affected (0.43 sec) 





图 13.8 命名 完整 性 约束 


a 
说 明 
在 定义 完整 性 约束 时 , 应 该 尽 可 能 为 其 指定 名 字 , 以 便 在 需要 对 完整 性 约束 进行 修改 或 删除 时 ， 
可 以 很 容易 地 找到 它们 。 


只 能 给 基于 表 的 完整 性 约束 指定 名 字 ， 无 法 给 基于 列 的 完整 性 约束 指定 名 字 。 


例 13.9 在 创建 表 时 添加 命名 外 键 完 整 性 约束 。〈 实 例 位 置 ; 光盘 \TMNsIN13\13.9) 
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在 本 实例 中 ,首先 创 建 一 个 图 书 类 别 信息 表 , 然后 再 创建 一 个 图 书信 息 表 ， 并 为 图 书信 息 表 设置 命名 
外 键 约束 ， 实 现 删 除 参 照 表 中 的 数据 时 ， 级 联 删 除 图 书信 息 表 中 相关 类 别 的 图 书信 息 ， 具 体 步 又 如 下 。 
(1) 创建 名 称 为 tb_type 的 图 书 类 别 信息 表 ， 具 体 代码 如 下 。 
CREATE TABLE tb_type ( 
id int(11) NOT NULL AUTO_INCREMENT, 


name varchar(45) DEFAULT NULL, 
remark varchar(100) DEFAULT NULL, 








PRIMARY KEY (id) 
有 


(2) 创建 不 添加 任何 外 键 的 教材 信息 表 tb_book， 代 码 如 下 。 


Create table tb_book(id int(11) not null primary key auto_increment, 
name varchar(20) not null, 
publishingho varchar(20) not null, 
author varchar(20), 

typeid int(11), 

CONSTRAINT fk_typeid 
FOREIGN KEY (typeid) 
REFERENCES tb_type(id) 

ON DELETE CASCADE 

ON UPDATE CASCADE 

由 


运行 结果 如 图 13.9 所 示 。 


db_database13; 
changed 

REATE TABLE th_type 《 

id intC11> NOT NULL AUTO_INCREMENT, 
name varcharC45> DEFAULT NULL., 
remark har(199》 DEFAULT NULL, 


PRIMARY KEY Cid*> 
> 2; 


JQuery OK. @ rows affected (0.38 sec) 


reate table th_hookCid intC11> not null prinary key auto_increment, 
name varch not null, 


publis 


> author 


RAINT fk_typeid 
yy Ctypeid) 


Ok. @ rows affected (8.34 sec) 





图 13.9 在 创建 表 时 添加 命名 外 键 完整 性 约束 
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13.3 更 新 完整 性 约束 


对 各 种 约束 命名 后 ， 就 可 以 使 用 ALTER TABLE 语句 来 更 新 或 删除 与 列 或 表 有 关 的 各 种 约束 。 下 
面 将 分 别 进行 介绍 。 





13.3.1 删除 完整 性 约束 


在 MySQL 中 ， 使 用 ALTER TABLE 语句 ， 可 以 独立 地 删除 完整 性 约束 ， 而 不 会 删除 表 本 身 。 如 
果 使 用 DROP TABLE 语句 删除 一 个 表 , 那么 这 个 表 中 的 所 有 完整 性 约束 也 会 自动 被 删除 。 删 除 完 整 性 
约束 需要 在 ALTER TABLE 语句 中 使 用 DROP 关键 字 来 实现 ， 有 具体 的 语法 格式 如 下 。 

DROP [FOREIGN KEY| INDEX| <symbol>] |[PRIMARY KEY] 

参数 说 明 如 下 。 

(1) FOREIGN KEY: 用 于 删除 外 键 约束 。 

(2) PRIMARY KEY: 用 于 删除 主键 约束 。 需 要 注意 的 是 ， 在 删除 主键 时 ， 必 须 再 创建 一 个 主键 ， 
否则 不 能 删除 成 功 。 

(3) INDEX: 用 于 删除 候选 键 约束 。 

(4) symbol: 要 删除 的 约束 名 称 。 

例 13.10 ”要 删除 例 13.8 中 的 名 称 为 信 _ classid 的 外 键 约束 ， 可 以 使 用 下 面 的 代码 。《〈 实 例 位 置 : 
光盘 \TMsIN13\13.10) 

ALTER TABLE tb_student1a DROP FOREIGN KEY fk_classid; 


运行 上 述 代 码 ， 其 结果 如 图 13.10 所 示 。 





图 13.10 删除 名 称 为 你 _classid 的 外 键 约束 


13.3.2 ”修改 完整 性 约束 


在 MySQL 中 ， 完 整 性 约束 不 能 直接 被 修改 ， 若 要 修改 只 能 使 用 ALTER TABLE 语句 先 删除 除 该 
约束 ， 然 后 再 增加 一 个 与 该 约束 同名 的 新 约束 。 由 于 删除 完整 性 约束 的 语法 在 13.3.1 节 已 经 介绍 了 ， 
这 里 只 给 出 在 ALTER TABLE 语句 中 添加 完整 性 约束 的 语法 格式 ， 具 体 语法 格式 如 下 。 

ADD CONSTRAINT <symbol> 各 种 约束 
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参数 说 明 如 下 。 

(1) symbol: 为 要 添加 的 约束 指定 一 个 名 称 。 

(2) 各 种 约束 : 定义 各 种 约束 的 语句 , 具体 内 容 请 参见 13.1 和 13.2 节 介绍 的 各 种 约束 的 添加 语法 。 

例 13.11 更 新 例 13.8 中 的 名 称 为 储 _classid 的 外 键 约束 为 级 联 删除 和 级 联 更 新 , 可 以 使 用 下 面 的 
代码 。〔 实 例 位 置 ， 光盘 \TMsN13\13.11) 





ALTER TABLE tb_student1a DROP FOREIGN KEY fk_classid; 
ALTER TABLE tb_student1a 

ADD CONSTRAINT fk_classid FOREIGN KEY (classid) 
REFERENCES tb_class(id) 

ON DELETE CASCADE 

ON UPDATE CASCADE 


运行 上 述 代 码 ， 其 结果 如 图 13.11 所 示 。 


auto_incrt 
《39》 not null, 


s _ affected (8- 


dentla DROP FOREIGN KEY fk_cl 


> ALTER TABLE thb_studentla 
TRAINT 上 d FOREIGN KEY 《classid> 


。B rows affected (0.58 
Duplicates: 0 Warning 





图 13.11 更 新 外 键 约束 


13.4 小 结 


本 章 主 要 介绍 了 定义 
的 束 和 命名 
据 库 设计 中 非常 实用 。 





性 约束 等 内 容 。 其 
完整 性 约束 是 本 章 的 重点 ， 需 要 读者 认真 学 习 、 灵 活 掌握 ， 这 在 以 后 的 数 
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13.5 ”实践 与 练习 


1. 在 创建 用 户 信息 表 tb_ manager 时 ， 将 id 字段 设置 为 主键 。 (答案 位 置 ; 光盘 \TMNsIM3\13.12) 
2. 创建 一 个 不 添加 任何 外 键 的 教师 信息 表 tb _teacher， 然 后 通过 ALTER TABLE 语句 为 其 添加 一 个 
名 称 为 储 _departmentid 的 外 键 约 束 。 (答案 位 置 : 光盘 \TMsIN13\13.13) 
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( a" 视频 讲解 ;: 22 分钟 ) 


存储 过 程 和 存储 函数 是 在 数据 库 中 定义 一 些 SQL 语句 的 集合 ,然后 直接 调用 这 
些 存 储 过 程 和 存储 函数 来 执行 已 经 定义 好 的 SQL 语句 ,可 以 避免 开发 人 员 重 复 编写 
相同 的 SQL 语句。 而 且 , 存储 过 程 和 存储 函数 是 在 MySQL 服务 器 中 存储 和 执行 的 ， 
可 以 减少 客户 端 和 服务 器 端的 数据 传输 。 本 章 将 介绍 存储 过 程 和 存储 函数 的 含义 、 
作用 ， 以 及 创建 、 使 有 用、 查看、 修改 及 删除 存储 过 程 和 存储 函数 的 方法 。 

通过 阅读 本 章 ， 读 者 可 以 : 

Wm 掌握 MySQL 中 存储 过 程 和 存储 困 数 的 创建 方法 

WI 掌握 存储 过 程 和 国 数 的 调用 、 查 看 、 修 改 和 删除 的 方法 
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14.1 创建 存储 过 程 和 存储 函数 


在 数据 库 系统 中 ， 为 了 保证 数据 的 完整 性 、 一 致 性 ， 同 时 也 为 提高 其 应 用 性 能 ， 大 多 数据 库 常 采 
用 存储 过 程 和 存储 函数 技术 。MySQL 在 5.0 版 本 后 ， 也 应 用 了 存储 过 程 和 存储 函数 。 存 储 过 程 和 存储 
函数 经 常 是 一 组 SQL 语句 的 组 合 ， 这 些 语句 被 当 作 整体 存 入 MySQL 数据 库 服务 器 中 。 用 户 定义 的 存 
储 函 数 不 能 用 于 修改 全 局 库 状 态 ， 但 该 函数 可 从 查询 中 被 唤醒 调用 ， 也 可 以 像 存 储 过 程 一 样 通过 语 扣 
执行 。 随 着 MySQL 技术 的 日 趋 完善 ， 存 储 过 程 将 和 存储 函数 在 以 后 的 项 目 中 得 到 广泛 的 应 用 。 


14.1.1 创建 存储 过 程 





在 MySQL 中 ， 创 建 存 储 过 程 的 基本 形式 如 下 。 


CREATE PROCEDURE sp_name ([proc_parameter[-]) 
[characteristic …] routine_body 


其 中 ，sp_name 参数 是 存储 过 程 的 名 称 ; proc_parameter 表示 存储 过 程 的 参数 列表 ; characteristic 
参数 指定 存储 过 程 的 特性 ，routine_body 参数 是 SQL 代码 的 内 容 ， 可 以 用 BEGIN…END 来 标识 SQL 
代码 的 开始 和 结束 。 


SO 
proc_parameter 中 的 参数 由 3 部 分 组 成 ， 它 们 分 别 是 输入 输出 类 型 、 参 数 名 称 和 参数 类 型 。 其 
形式 为 [IN| OUT |INOUT ]param name type. 其 中 , IN 表示 输入 参数 ; OUT 表示 输出 参数 ; INOUT 
表示 既 可 以 输入 也 可 以 输出 ; param name 参数 是 存储 过 程 参数 名 称 ; type 参数 指定 存储 过 程 的 参 
数 类 型 ， 该 类 型 可 以 为 MySQL 数据 库 的 任意 数据 类 型 。 


一 个 存储 过 程 包括 名 字 、 参 数列 表 ， 还 可 以 包括 很 多 SQL 语句 集 。 下 面 创建 一 个 存储 过 程 ， 其 代 
码 如 下 。 

delimiter // 

create procedure proc_name (in parameter integer) 

begin 

declare variable varchar(20); 

if parameter=1 then 

set variable='MySQL '; 

else 

set variable='PHP"; 

end 所 

insert into tb (name) values (variable); 

end; 
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MySQL 中 存储 过 程 的 建立 以 关键 字 create procedure 开始 ， 后 面 紧 跟 存 储 过 程 的 名 称 和 参数 。 
MySQL 的 存储 过 程 名 称 不 区 分 大 小 写 ， 如 PROCE10 和 proce10 代 表 同 一 存储 过 程 名 。 存 储 过 程 名 或 
存储 函数 名 不 能 与 MySQL 数据 库 中 的 内 建 函 数 重 名 。 

MySQL 存储 过 程 的 语句 块 以 begin 开始 ， 以 end 结束 。 语句 体 中 可 以 包含 变量 的 声明 、 控 制 语句 、 
SQL 查询 语句 等 。 由 于 存储 过 程 内 部 语句 要 以 分 号 结束 ， 所 以 在 定义 存储 过 程 前 ， 应 将 语句 结束 标志 
“;” 更 改 为 其 他 字符 ， 并 且 应 降低 该 字符 在 存储 过 程 中 出 现 的 机 率 ， 更 改 结束 标志 可 以 用 关键 字 
delimiter 定义 ， 例 如 : 











mysql>delimiter // 
存储 过 程 创建 之 后 ， 可 用 如 下 语句 进行 删除 ， 参 数 proc_name 指 存储 过 程 名 。 
drop procedure proc_name 
下 面 创 建 一 个 名 称 为 count_of student 的 存储 过 程 。 首 先 ， 创 建 一 个 名 称 为 students 的 MySQL 数 
据 库 ， 然 后 创建 一 个 名 为 studentinfo 的 数据 表 。 数 据 表 结构 如 表 14.1 所 示 。 
表 14.1 studentinfo 数据 表 结 构 


字段 名 说 明 
ee |  。 主 负 和 型 别 
age | vagcHa | | | 4 和 4 


ex varciap®) | M | | 尖 便 
a aemay | | | ke 


例 14.1 创建 一 个 名 称 为 count_of student 的 存储 过 程 ,统计 studentinfo 数据 表 中 的 记录 数 。 代 码 
如 下 。《〈 实 例 位 置 : 光盘 \TMNsN14\14.1) 





delimiter // 

create procedure count_of_student(OUT count_num INT) 
reads sql data 

begin 

select count(*) into count_num from studentinfo; 

end 

UA 


在 上 述 代码 中 ， 定 义 一 个 输出 变量 count num， 存 储 过 程 应 用 SELECT 语句 从 studentinfo 表 中 获 
取 记 录 总 数 ， 最 后 将 结果 传递 给 变量 count_ num。 存 储 过 程 的 执行 结果 如 图 14.1 所 示 。 





图 14.1 创建 存储 过 程 count_of student 
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代码 执行 完毕 后 ， 没 有 报 出 任何 出 错 信 息 就 表示 存储 函数 已 经 创建 成 功 。 以 后 就 可 以 调用 这 个 存 
Mt 数据 库 中 会 执行 存储 过 程 中 的 SQL 语句 。 
避税 曙 
MySQL 中 默认 的 语句 结束 符 为 分 号 ; 存储 过 程 中 的 SQL 语句 需要 分 号 来 结束 .为 了 避免 冲突 ， 
首先 用 “DELIMITER //” 将 MySQL 的 结束 符 设置 为 /， 最 后 再 用 “DELIMITER;” 来 将 结束 符 恢 
复 成 分 号 。 这 与 创建 触发 器 时 是 一 样 的 。 


14.1.2 ”创建 存储 函数 


创建 存储 函数 与 创建 存储 过 程 大 体 相 同 ， 创 建 存储 函数 的 基本 形式 如 下 。 


CREATE FUNCTION sp_name ([func_parameter[,…]]) 
RETURNS type 
[characteristic …] routine_body 
创建 存储 函数 的 参数 说 明 如 表 14.2 所 示 。 
表 14.2 创建 存储 函数 的 参数 说 明 


参数 说 明 
sp_name 存储 函数 的 名 称 
fun_parameter 存储 函数 的 参数 列表 
RETURNS type 指定 返回 值 的 类 型 
characteristic 指定 存储 过 程 的 特性 
Toutine body SQL 代码 的 内 容 


func_parameter 可 以 由 多 个 参数 组 成 ， 其 中 每 个 参数 均 由 参数 名 称 和 参数 类 型 组 成 ， 其 结构 如 下 。 

param_name type 

param_name 参数 是 存储 函数 的 函数 名 称 ; type 参数 用 于 指定 存储 函数 的 参数 类 型 。 该 类 型 可 以 是 
MySQL 数据 库 所 支持 的 类 型 。 

例 14.2 同样 ， 应 用 studentinfo 表 ， 创建 名 为 name_of student 的 存储 函数 ， 其 代码 如 下 。 (实例 
位 置 : 光盘 \TMNsIN14\14.2) 

delimiter // 

create function name_of_student(std_id INT) 

returns varchar(50) 

begin 

return(select name from studentinfo where sid=std_id); 


end 
I 
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上 述 代 码 中 ， 存 储 函 数 的 名 称 为 name_of student; 该 函数 的 参数 为 std id; 返回 值 是 VARCHAR 
类 型 。 该 函数 实现 从 studentinfo 表 查 询 与 std_ id 相同 sid 值 的 记录 ， 并 将 学 生 名 称 字段 name 中 的 值 返 
本。 存储 函数 的 执行 结果 如 图 14.2 所 示 。 
































9 OK. 0 rows affected (0.83 sec) 





图 14.2 创建 name_of student0 存 储 函数 
14.1.3 ”变量 的 应 用 


MySQL 存储 过 程 中 的 参数 主要 有 局 部 参数 和 会 话 参 数 两 种 , 这 两 种 参数 又 可 以 被 称 为 局 部 变量 和 
会 话 变 量 。 局 部 变量 只 在 定义 该 局 部 变量 的 begin…end 范围 内 有 效 ， 会 话 变量 在 整个 存储 过 程 范 围 内 
均 有 效 。 


1. 局 部 变量 

局 部 变量 以 关键 字 DECLARE 声明 ， 后 跟 变 量 名 和 变量 类 型 ， 例 如 : 

declare a int 

当然 ， 在 声明 局 部 变量 时 也 可 以 用 关键 字 DEFAULT 为 变量 指定 默认 值 ， 例 如 : 

declare a int default 10 

下 述 代码 为 读者 展示 如 何在 MySQL 存储 过 程 中 定义 局 部 变量 以 及 其 使 用 方法 。 在 该 例 中 ， 
在 内 层 和 外 层 BEGIN…END 块 中 都 定义 同名 的 变量 x, 按照 语句 从 上 到 下 执行 的 顺序 ， 如 果 变 量 x 有 
整个 程序 中 都 有 效 , 则 最 终结 果 应 该 都 为 inner, 但 真正 的 输出 结果 却 不 同 , 这 说 明 在 内 部 BEGIN…END 
块 中 定义 的 变量 只 在 该 块 内 有 效 。 


例 14.3 本 例 说 明 局 部 变量 只 在 某 个 BEGIN…END 块 内 有 效 ， 代 码 如 下 。 (实例 位 置 ; 光盘 
VTMsN14\14.3) 


delimiter // 

create procedure p1() 

begin 

declare x char(10) default 'outer '; 
begin 

declare x char(10) default 'inner '; 
select x; 

end; 

select x; 

end; 

I 
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上 述 代 码 的 运行 结果 如 图 14.3 所 示 。 


> deliniter // 
ate procedure piC) 


x charC18> default ’outer ’; 


19》 default ’inner 





图 14.3 ”定义 局 部 变量 的 运行 结果 
应 用 MySQL 调用 该 存储 过 程 的 运行 结果 如 图 14.4 所 示 。 


hysql> call p1C)// 


in set 8.83 sec) 


Ok. Ox 


图 14.4 调用 存储 过 程 pl0 的 运行 结果 





2. 全 局 变量 


MySQL 中 的 会 话 变量 不 必 声 明 即 可 使 用 ， 会 话 变量 在 整个 过 程 中 有 效 ， 会 话 变量 名 以 字符 “@” 
作为 起 始 字符 

例 14.4 分别 在 内 部 和 外 部 BEGIN…END 块 中 都 定义 了 同名 的 会 话 变量 @t， 并 且 最 终 输 出 结果 
相同 ， 从 而 说 明 会 话 变量 的 作用 范围 为 整个 程序 。 设 置 全 局 变量 的 代码 如 下 。 【实例 位 置 ， 光盘 
VIMsN14\14.4) 


delimiter // 
create procedure p2() 
begin 

set @t=1; 
begin 

set @t=2; 
select @t; 
end; 
select @t; 
end; 

I 


上 述 代 码 的 运行 结果 如 图 14.5 所 示 。 
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> deliniter / 
e procedure p2C) 


Query OK. @ rows affected (@.80 sec) 





图 14.5 设置 全 局 变量 
应 用 MySQL 调用 该 存储 过 程 的 运行 结果 如 图 14.6 所 示 。 


Inysql> call p2C)// 
[3233 


1 row in set (8.81 sec》 


1 et 


1 row in set (0.01 sec》 


lQuery OK. @ rows affected 〔 





图 14.6 调用 存储 过 程 p20 运行 结果 

3. 为 变量 赋值 

MySQL 中 可 以 使 用 关键 字 DECLARE 来 定义 变量 ， 其 基本 语法 如 下 。 

DECLARE var_name[,…] type [DEFAULT value] 

DECLARE 是 用 来 声明 变量 的 ，var_name 参数 是 设置 变量 的 名 称 。 如 果 用 户 需 要 ， 也 可 以 同时 定 
义 多 个 变量 ;type 参数 用 来 指定 变量 的 类 型 ， DEFAULT value 的 作用 是 指定 变量 的 默认 值 ， 不 对 该 参 
数 进 行 设置 时 ， 其 默认 值 为 NULL。 

MySQL 中 可 以 使 用 关键 字 SET 为 变量 赋值 ， 其 基本 语法 如 下 。 

SET var_name=expr[,var_name=expr] … 

SET 关键 字 是 用 来 为 变量 赋值 ，var_name 参数 是 变量 的 名 称 ; expr 参数 是 赋值 表达 式 。 一 个 SET 
语句 可 以 同时 为 多 个 变量 赋值 ， 各 个 变量 的 赋值 语句 之 间 用 “,” 隔 开 。 例 如 ， 为 变量 mr_soft 赋值， 
代码 如 下 。 

SET mr_soft=10; 

另外 ，MySQL 中 还 可 以 应 用 另 一 种 方式 为 变量 赋值 ， 其 语法 结构 如 下 。 

SELECT col_name[,…] INTO var_namel,…] FROM table_name where condition 
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其 中 ，col_name 参数 标识 查询 的 字段 名 称 ，var_name 参数 是 变量 的 名 称 ; table_name 参数 为 指定 
数据 表 的 名 称 ，condition 参数 为 指定 查询 条 件 。 例 如 ， 从 studentinfo 表 中 查询 name 为 “LeonSK” 的 
记录 ， 并 将 该 记录 下 的 tel 字段 内 容 赋值 给 变量 customer_tel， 其 关键 代码 如 下 。 


SELECT tel INTO customer_tel FROM studentinfo WHERE name= 'LeonSK ': 


/ 
CO 说 明 
上 述 赋值 语句 必须 存在 于 创建 的 存储 过 程 中 ， 且 需 将 赋值 语句 放置 在 BEGIN-…END 之 间 。 若 
脱离 此 范围 ， 该 变量 将 不 能 使 用 或 被 赋值 。 


14.1.4 ”光标 的 运用 


通过 MySQL 查询 数据 库 ， 其 结果 可 能 为 多 条 记录 。 在 存储 过 程 和 函数 中 使 用 光标 可 以 实现 逐条 
读 取 结 果 集中 的 记录 。 光 标 使 用 包括 声明 光标 (DECLARE CURSOR) 、 打 开光 标 (OPEN CURSOR) 、 
使 用 光标 (FETCH CURSOR) 和 关闭 光标 ‘CLOSE CURSIR) 。 值 得 一 提 的 是 ， 光 标 必 须 声 明 在 处 理 
程序 之 前 ， 且 声明 在 变量 和 条 件 之 后 。 

1. 声明 光标 

在 MySQL 中 ， 声 明光 标 仍 使 用 关键 字 DECLARE， 其 语法 如 下 。 

DECLARE cursor_name CURSOR FOR select_statement 

其 中 ，cursor name 是 光标 的 名 称 ， 光 标 名 称 使 用 与 表 名 同样 的 规则 ; select_statement 是 一 个 
SELECT 语句 ， 返 回 一 行 或 多 行 数据 。 该 语句 也 可 以 在 存储 过 程 中 定义 多 个 光标 ， 但 是 必须 保证 每 个 
光标 名 称 的 唯一 性 ， 即 每 一 个 光标 必须 有 自己 唯一 的 名 称 。 

通过 上 述 定义 来 声明 光标 info_of_ student， 其 代码 如 下 。 

DECLARE info_of_student CURSOR FOR SELECT 

sid,name,age,sex,age 


FROM studentinfo 
WHERE sid=1; 


/ 
说 明 
这 里 SELECT 子 句 中 不 能 包含 INTO 子 句 , 并 且 光 标 只 能 在 存储 过 程 或 存储 函数 中 使 用 。 上述 
代码 并 不 能 单独 执行 。 














2. 打开 光标 


在 声明 光标 之 后 ， 要 从 光标 中 提取 数据 ， 必 须 首先 打开 光标 。 在 MySQL 中 使 用 关键 字 OPEN 来 
打开 光标 ， 其 基本 的 语法 如 下 。 


OPEN cursor_name 
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其 中 ，cursor name 参数 表示 光标 的 名 称 。 在 程序 中 ， 一 个 光标 可 以 打开 多 次 。 由 于 可 能 在 用 户 打 
开光 标 后 ， 其 他 用 户 或 程序 正在 更 新 数据 表 ， 所 以 可 能 会 导致 用 户 在 每 次 打开 光标 后 ， 显 示 的 结果 都 
不 同 。 

打开 上 面 已 经 声明 的 光标 info_of student， 其 代码 如 下 。 

OPEN info_of_student 

3. 使 用 光标 

光标 在 顺利 打开 后 ， 可 以 使 用 FETCH…INTO 语句 来 读 取 数 据 ， 其 语法 如 下 。 

FETCH cursor_name INTO var_name[var_name]… 

其 中 , cursor_name 代表 已 经 打开 光标 的 名 称 ; var_name 参数 表示 将 光标 中 的 SELECT 语句 查询 出 
来 的 信息 存 入 该 参数 中 。var_name 是 存放 数据 的 变量 名 ， 必 须 在 声明 光标 前 定义 好 。FETCH…INTO 
语句 与 SELECT…INTO 语句 具有 相同 的 意义 。 

将 已 打开 的 光标 info_of student 中 SELECT 语句 查询 出 来 的 信息 存 入 tmp_name 和 tmp tel 中。 其 
中 ，tmp_name 和 tmp_tel 必须 在 使 用 前 定义 。 其 代码 如 下 。 

FETCH info_of_student INTO tmp_name,tmp_tel; 

4. 关闭 光标 

光标 使 用 完毕 后 ， 要 及 时 关闭 ， 在 MySQL 中 采用 关键 字 CLOSE 关闭 光标 ， 其 语法 格式 如 下 。 

CLOSE cursor_name 

cursor_name 参数 表示 光标 名 称 。 下 面 关闭 已 打开 的 光标 info_of student， 其 代码 如 下 。 

CLOSE info_of_student 


/ 
和 培 明 
对 于 已 关闭 的 光标 ， 在 其 关闭 之 后 则 不 能 使 用 关键 字 FETCH 来 使 用 光标 。 光 标 在 使 用 完毕 后 
一 定 要 关闭 。 

















14.2 ”存储 过 程 和 存储 函数 的 调用 


存储 过 程 和 存储 函数 都 是 存储 在 服务 器 的 SQL 语句 的 集合 。 要 使 用 这 些 已 经 定义 好 的 存储 过 程 和 存储 
函数 就 必须 要 通过 调用 的 方式 来 实现 。 对 存储 过 程 和 函数 的 操作 主要 可 以 分 为 调用 、 查 看 、 修 改 和 删除 。 


14.2.1 调用 存储 过 程 
存储 过 程 的 调用 在 前 面 的 示例 中 多 次 被 用 到 。MySQL 中 使 用 CALL 语句 来 调用 存储 过 程 。 调 用 存 
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储 过 程 后 ， 数 据 库 系 统 将 执行 存储 过 程 中 的 语句 ， 然 后 将 结果 返回 给 输出 值 。CALL 语句 的 基本 语法 
形式 如 下 。 
CALL sp_name([parameter[,…-]]); 


其 中 ，sp_name 是 存储 过 程 的 名 称 ; parameter 是 存储 过 程 的 参 才 














14.2.2 ”调用 存储 函数 


在 MySQL 中 , 存储 函数 的 使 用 方法 与 MySQL 内 部 函数 的 使 用 方法 基本 相同 。 用 户 自 定义 的 存储 
函数 与 MySQL 内 部 函数 性 质 相 同 。 区 别 在 于 , 存储 函数 是 用 户 自 定义 的 , 而 内 部 函数 由 MySQL 自 带 。 
其 语法 结构 如 下 。 

SELECT function_name([parameter[…-]); 

例 14.5 创建 存储 过 程 并 在 PHP 中 调用 该 存储 过 程 实现 用 户 注 册 。【〔 实 例 位 置 光盘 
\TMNsI\14\14.5) 

(1) 创建 pro_reg 存储 过 程 ， 其 代码 如 下 。 


delimiter // 

create procedure pro_reg(in nc varchar(50),in pwd varchar(50),in email varchar(50),in address varchar(50)) 
begin 

insert into tb_reg(name,pwd,email,address) values (nc,pwd,email,address); 

end; 

UA 


执行 效果 如 图 14.7 所 示 。 


nc varcharC50) ,in pwd varchar(50),in email va 


s nc.pwd,enail,add 


Query OK, @ rows affected G0.02 sec) 





图 14.7 创建 存储 过 程 
(2) 通过 PHP 预定 义 类 mysqli 实现 与 MySQL 数据 库 的 连接 ， 代 码 如 下 。 


<?php 
if($_POST[submit]!="X{ 
$conn=new mysqli("localhost","root","root","db_database14"); /| 连接 数据 库 
S$conn->query("set names utf8"); // 设 置 编码 格式 


$name=$_POST[name]; 
$pwd=md5($_POST[pwd]); 
$email=$_POST[email]; 
$address=$_POST[address'; 
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(3) 调用 存储 过 程 pro_reg 实现 将 用 户 录入 的 注册 信息 保存 到 数据 库 ， 代 码 如 下 。 


if($sql=$conn->query("call pro_reg(".$name.",".$pwd.",".$email.",".$address.")")\{ ”// 调 用 存储 过 程 
echo "<script>alert(' 用 户 注册 成 功 ");</script>"; 
jelse{ 
echo "<script>alert(' 用 户 注册 失败 !");</script>"; 
b 
} 


?> 


运行 本 实例 ， 在 文本 框 中 输入 如 图 14.8 所 示 的 注册 信息 后 ， 单 击 “注册 ”按钮 即 可 将 用 户 填写 的 
注册 信息 保存 到 数据 库 中 ， 最 终 保 存 结果 如 图 14.9 所 示 。 


' 及 任 续 论坛 
HTTEIIMINGRI SOFT COM 阴 目 树 近 












































用 户 注册 | 
用 户 昵称 : mrlzh 
注册 密码 : [IIIIIII 
E-mail: jlru_1zht*##@163. com | 
家 庭 住址 : [吉林 长 者 | 
注册 | | 重 写 











图 14.8 录入 注册 信息 


一， id name pwd email address 
DX 1 mh 25d55ad283aa400af464c76d713c07ad jinu_lzh™™@163.com 吉林 长 春 


图 14.9 注册 信息 被 存储 到 MySQL 数据 库 
14.3 ”查看 存储 过 程 和 存储 函数 


存储 过 程 和 存储 函数 创建 以 后 ， 用 户 可 以 查看 存储 过 程 和 存储 函数 的 状态 和 定义 。 用 户 可 以 通过 
SHOW STATUS 语句 查看 存储 过 程 和 存储 函数 状态 ， 也 可 以 通过 SHOW CREATE 语句 来 查看 存储 过 
程 和 存储 函数 的 定义 。 


14.3.1 SHOW STATUS 语句 


在 MySQL 中 可 以 通过 SHOW STATUS 语句 查看 存储 过 程 和 存储 函数 的 状态 。 其 基本 语法 结构 
如 下 。 


SHOW {PROCEDURE | FUNCTION}STATUSILIKE 'pattem] 
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其 中 ，PROCEDURE 参数 表示 查询 存储 过 程 ; FUNCTION 参数 表示 查询 存储 函数 ;LIKE 'pattern' 
参数 用 来 匹配 存储 过 程 或 存储 函数 名 称 。 





14.3.2 SHOW CREATE 语句 


MySQL 中 可 以 通过 SHOW CREATE 语句 来 查看 存储 过 程 和 函数 的 状态 下 法 结果 如 下 。 
SHOW CREATE{PROCEDURE | FUNCTION } sp_name; 





其 中 ，PROCEDURE 参数 表示 存储 过 程 ， FUNCTION 参数 表示 查询 存储 函数 ，sp_name 参数 表示 
存储 过 程 或 函数 的 名 称 。 
例 14.6 下 面 查 询 名 为 count_ of student 的 存储 过 程 , 其 代码 如 下 。.( 实 例 位 置 :光盘 \TMNsI\14\14.6) 


show create procedure count_of_student ; 


其 运行 结果 如 图 14.10 所 示 。 





! Procedure HE ! e Procedure 


_student ! NO_AUTO_CREATE_USER ! CREATE DEFINER=“root“@"“localhost. PRO 
nt_of _student 《COUT count_num INT> 





elect countC*> into count_num from studentinfo; 


图 14.10 应 用 SHOW CREATE 语句 查看 存储 过 程 
pn :存储 过 程 的 定义 、 字 符 集 等 信息 。 
说 明 
SHOW STATUS 语句 只 能 查看 存储 过 程 或 函数 所 操作 的 数据 库 对 象 ， 如 存储 过 程 或 函数 的 名 
称 、 类 型 、 定 义 者 、 修 改 时 间 等 信息 ， 并 不 能 查询 存储 过 程 或 函数 的 具体 定义 。 如 果 需 要 查看 详细 
定义 ， 需 要 使 用 SHOW CREATE 语句 。 


14.4 修改 存储 过 程 和 存储 函数 


修改 存储 过 程 和 存储 函数 是 指 修改 已 经 定义 好 的 存储 过 程 和 函数 。MySQL 中 通过 ALTER 
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PROCEDURE 语句 来 修改 存储 过 程 ， 通 过 ALTER FUNCTION 语句 来 修改 存储 函数 。 
MySQL 中 修改 存储 过 程 和 存储 函数 的 语句 的 语法 形式 如 下 。 


ALTER {PROCEDURE | FUNCTION} sp_name [characteristic -..] 
characteristic: 
{ CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA} 
| SQL SECURITY { DEFINER | INVOKER } 
| COMMENT 'string' 


其 参数 说 明 如 表 14.3 所 示 。 
表 14.3 ”修改 存储 过 程 和 存储 函数 的 语法 的 参数 说 明 























参数 说 明 
sp_name 存储 过 程 或 函数 的 名 称 
characteristic 指定 存储 函数 的 特性 
CONTAINS SQL 表示 子 程序 包含 SQL 语句 ， 但 不 包含 读 写 数 据 的 语句 
NO SQL 表示 子 程序 不 包含 SQL 语句 
READS SQL DATA 表示 子 程序 中 包含 读数 据 的 语句 
MODIFIES SQL DATA 表示 子 程序 中 包含 写 数 据 的 语句 
SQL 指明 权限 执行 。 DEFINER 表示 只 有 定义 者 自己 才能 够 执行 ， INVOKER 表 
SECURITY {DEFINERIINVOKER} 示 调 用 者 可 以 执行 
COMMENT 'string' 是 注释 信息 


例 14.7 下 面 应 用 此 语句 修改 存储 过 程 count_of student， 其 代码 如 下 。〔 实 例 位 置 ， 光盘 
\TMNsN\14\14.7) 


alter procedure count_of_student 
modifies sql data 
sql security invoker; 


其 运行 结果 如 图 14.11 所 示 。 


> alter procedure count_of_student 


> nodifies sql data 


ql security invoker; 


- 日 rows affected (0@.80 sec) 





图 14.11 修改 存储 过 程 count_of student 的 定义 


ma 
说 明 
如 果 读 者 希望 查看 修改 后 的 结果 。 可 以 应 用 SELECT…FROM studentinfo Ruotines WHERE 
ROUTINE NAME='sp_name' 来 查看 表 的 信息 。 由 于 篇 幅 限 制 ， 这 里 不 进行 详细 讲解 。 
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14.5 删除 存储 过 程 和 存储 函数 


删除 存储 过 程 和 存储 函数 指 删除 数据 库 中 已 经 存在 的 存储 过 程 或 存储 函数 。MySQL 中 通过 DROP 
PROCEDURE 语句 来 删除 存储 过 程 ， 通 过 DROP FUNCTION 语句 来 删除 存储 函数 。 在 删除 之 前 ， 必 须 
确认 该 存储 过 程 或 函数 没有 任何 依赖 关系 ， 否 则 可 能 会 导致 其 他 与 其 关联 的 存储 过 程 无 法 运行 。 

删除 存储 过 程 和 存储 函数 的 语法 如 下 。 

DROP {PROCEDURE | FUNCTION} [IF EXISTS] sp_name 

其 中 ，sp_name 参数 表示 存储 过 程 或 存储 函数 的 名 称 ; 下 EXISTS 是 MySQL 的 扩展 ,判断 存储 过 
程 或 函数 是 否 存在 ， 以 免 发 生 错误 。 

例 14.8 下 面 删除 名 称 为 count_of student 的 存储 过 程 ， 其 关键 代码 如 下 。〔 实 例 位置 ， 光 盘 
\TMNsN\14\14.8) 

drop procedure count_of_student; 


删除 存储 过 程 count_of student 的 运行 结果 如 图 14.12 所 示 。 





图 14.12 ”删除 count of student 存储 过 程 


例 14.9 下 面 删 除名 称 为 name_of student 的 存储 函数 ， 其 关键 代码 如 下 。《【 实 例 位 置 ， 光盘 
\TMN\s\14\14.9) 


drop function name_of_student; 


删除 存储 函数 name_of student 的 运行 结果 如 图 14.13 所 示 。 


of_student 


d 《B.B8 sec) 





图 14.13 ”删除 name_of student 存储 函数 


当 返 回 结果 没有 提示 警告 或 报错 时 ， 则 说 明 存 储 过 程 或 存储 函数 已 经 被 顺利 删除 。 用 户 可 以 通过 
查询 students 数据 库 下 的 Routines 表 来 确认 上 面 的 删除 是 否 成 功 。 





14.6 小 3 








本 章 对 MySQL 数据 库 的 存储 过 程 和 存储 函数 进行 了 详细 讲解 ， 存 储 过 程 和 存储 函数 都 是 用 户 自 
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己 定义 的 SQL 语句 的 集合 。 它 们 都 存储 在 服务 器 端 ， 只 要 调用 就 可 以 在 服务 器 端 执行 。 本 章 重点 讲解 
了 创建 存储 过 程 和 存储 函数 的 方法 。 通过 CREATE PROCEDURE 语句 来 创建 存储 过 程 , 通过 CREATE 
FUNCTION 语句 来 创建 存储 函数 。 这 两 个 内 容 是 本 章 的 难点 ， 需 要 读者 将 书 中 的 知识 点 结合 实际 操作 
进行 练习 。 


14.7 实践 与 练习 


1. 将 名 称 为 name_of student 的 存储 函数 的 读 写 权限 修改 为 READS SQL DATA, 并 加 上 注释 信息 


“FIND NAME”。 (答案 位 置 : 光盘 \TMsIN14\14.10) 
2. 通过 SHOW STATUS 语句 查看 存储 函数 的 状态 。 (答案 位 置 : 光盘 \TMNsIN14\14.11) 


FRE 


触 发 器 


(后 ! 视频 讲解 : 22 分 钟 ) 


触发 器 是 由 事件 来 触发 基 个 操作 。 这 些 事件 包括 INSERT 语句 、UPDATE 语句 
和 DELETE 语句 。 当 数据 库 系 统 执行 这 些 事件 时 , 就 会 激活 触发 器 执行 相应 的 操作 。 
本 章 将 对 触发 器 的 含义 、 作 用 ， 以 及 创建 、 查 看 和 删除 触发 器 的 方法 进行 详细 介绍 。 
通过 阅读 本 章 ， 读 者 可 以 : 


ml 


ed 


pl 


了 解 触发 路 的 概念 

了 解 创建 单条 执行 语句 的 触发 路 的 方法 
掌握 创建 多 条 执行 语句 的 触发 器 的 方法 
掌握 查看 触发 中 的 方法 
掌握 使 用 触发 中 的 方法 
掌握 删除 触发 路 的 方法 
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15.1 MySQL 触发 器 


触发 器 是 由 MySQL 的 基本 命令 事件 来 触发 某 种 特定 操作 , 这 些 基本 的 命令 由 INSERT、UPDATE、 
DELETE 等 事件 来 触发 某 些 特定 操作 。 满 足 触发 器 的 触发 条 件 时 ， 数 据 库 系统 就 会 自动 执行 触发 器 中 
定义 的 程序 语句 ， 可 以 令 某 些 操 作 之 间 的 一 致 性 得 到 协调 。 


15.1.1 创建 MySQL 触发 器 


在 MySQL 中， 创建 只 有 一 条 执行 语句 的 触发 器 的 基本 形式 如 下 。 


CREATE TRIGGER 触发 器 名 BEFORE | AFTER 触发 事件 
ON 表 名 FOR EACH ROW 执行 语句 


具体 的 参数 说 明 如 下 。 

(1) 触发 器 名 指定 要 创建 的 触发 器 名 字 。 

(2) 参数 BEFORE 和 AFTER 指定 触发 器 执行 的 时 间 。BEFORE 指 在 触发 时 间 之 前 执行 触发 语句 ; 
AFTER 表示 在 触发 时 间 之 后 执行 触发 语句 。 

(3) 触发 时 间 参 数 指数 据 库 操作 触发 条 件 ， 其 中 包括 INSERT、UPDATE 和 DELETE。 

(4) 表 名 指定 触发 时 间 操 作 表 的 名 称 。 

(5) FOR EACH ROW 表示 任何 一 条 记录 上 的 操作 满足 触发 事件 都 会 触发 该 触发 器 。 

(6) 执行 语句 指 触 发 器 被 触发 后 执行 的 程序 。 

例 15.1 下 面 创建 一 个 由 插入 命令 INSERT 触发 的 触发 器 auto_save_time， 具 体 步 骤 如 下 。 (实例 
位 置 ， 光盘 \TMNsIMS\15.1) 

(1) 创建 一 个 名 称 为 timelog 的 表格 ， 该 表 的 结构 非常 简单 。 相 关 代 码 如 下 所 示 。 

create table timelog( 

id int(11) primary key auto_increment not null, 
savetime varchar(50) not null 


}s 
(2) 创建 名 称 为 auto_save_time 的 触发 器 ， 其 代码 如 下 。 


delimiter // 

create trigger auto_save_time before insert 
on studentinfo for each row 

insert into timelog(savetime) values(now()); 
儿 


以 上 代码 的 运行 结果 如 图 15.1 所 示 。 
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auto_save_time 触发 器 创建 成 功 ， 其 具体 的 功能 是 当 用 户 向 studentinfo 表 中 执行 INSERT 操作 时 ， 
数据 库 系 统 会 自动 在 插入 语句 执行 之 前 向 timelog 表 中 插入 当前 时 间 。 下 面 通过 向 studentinfo 表 中 插入 
-条 信息 来 查看 触发 器 的 作用 ， 其 代码 如 下 所 示 。 
insert into studentinfo(name) values (Chris'); 


然后 执行 SELECT 语句 查看 timelog 表 中 是 否 执行 INSERT 操作 ， 其 结果 如 图 15.2 所 示 。 














> deliniter // 
create trigger auto_saue_time before insert 
studentinfo for each row 
rt into tinelogCsavetine) valuesnowC)); 


IlQuery OK, @ rows affected (0.80 sec» 





图 15.1 创建 auto_save_time 触发 器 图 15.2 查看 timelog 表 中 是 否 执行 插入 操作 


以 上 结果 显示 ， 在 向 studentinfo 表 中 插入 数据 时 ，savetime 表 中 也 会 被 插入 一 条 当前 系统 时 间 的 
数据 。 


15.1.2 创建 具有 多 条 执行 语句 的 触发 器 


15.1.1 节 中 已 经 介绍 了 如 何 创 建 一 个 最 基本 的 触发 器 , 但 是 在 实际 应 用 中 , 往往 触发 器 中 包含 多 条 
执行 语句 。 其 中 创建 具有 多 条 执行 语句 的 触发 

CREATE TRIGGER 触发 器 名 称 BEFORE | AFTER 触发 事件 

ON 表 名 FOR EACH ROW 

BEGIN 

执行 语句 列表 

END 

其 中 ,创建 具有 多 条 执行 语句 触发 器 的 语法 结构 与 创建 触发 器 的 一 般 语法 结构 大 体 相同 ， 其 参数 
说 明 请 参考 15.1.1 节 中 的 参数 说 明 ， 这 里 不 再 袭 述 。 在 该 结构 中 ， 将 要 执行 的 多 条 语句 放 入 BEGIN 与 
END 之 间 。 多 条 语句 需要 执行 的 内 容 ， 需 要 用 分 隔 符 “:” 隔 开 。 


说 明 

一 般 放 在 BEGIN 与 END 之 间 的 多 条 执行 语句 必须 用 结束 分 隔 符 “:” 分 开 。 在 创建 触发 器 过 
程 中 需要 更 改 分 隔 符 ， 故 这 里 应 用 第 14 章 提 到 的 DELIMITERT 语句 ， 将 结束 符号 变 为 “//”。 当 
触发 器 创建 完成 后 ， 读 者 同样 可 以 应 用 该 语句 将 结束 符 换 回 “;”。 








例 15.2 创建 一 个 由 DELETE 触发 多 条 执行 语句 的 触发 器 delete_time_info。 模拟 一 个 删除 日 志 数 
据 表 和 一 个 删除 时 间 表 。 当 用 户 删 除数 据 库 中 的 某 条 记录 后 ， 数 据 库 系统 会 自动 向 日 志 表 中 写 入 日 志 
信息 。 创 建 具有 多 个 执行 语句 的 触发 器 的 过 程 如 下 : 

在 例 15.1 中 创建 的 timelog 数据 表 基 础 上 , 另外 创建 一 个 名 称 为 timeinfo 的 数据 表 , 代码 如 下 。( 实 
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例 位 置 ， 光盘 \TMNsI\M5\15.2) 


create table timeinfo( 
id int(11) primary key auto_increment, 
info varchar(50) not null 


MI 
然后 创建 一 个 由 DELETE 触发 多 条 执行 语句 的 触发 器 delete_time_info， 其 代码 如 下 。 
delimiter // 


create trigger delete_time_info after delete 
on studentinfo for each row 

begin 

insert into timelog(savetime) values (now()); 
insert into timeinfo(info) values ('deleteact ); 
end 

I 


lQuery Ok. ss affected (@.85 sec) 





图 15.3 创建 具有 多 个 执行 语句 的 触发 器 delete_time_info 
例 15.3 触发 器 创建 成 功 ， 当 执行 删除 操作 后 ，timelog 与 timeinfo 表 中 将 会 插入 两 条 相关 记录 。 


执行 删除 操作 的 代码 如 下 。《 实 例 位 置 : 光盘 \TMNsN\15\15.3》 


DELETE FROM studentinfo where sid=7; 


删除 成 功 后 ， 应 用 SELECT 语句 分 别 查 看 数据 表 timelog 与 数据 表 timeinfo， 其 运行 结果 如 图 15.4 


和 图 15.5 所 示 。 


Inysql> select * fron tineinfo // 
! ia 1 info 
1 ! deleteact ! 


1 row in set Cg.00 sec) 





图 15.4 查看 数据 表 timelog 信息 图 15.5 查看 数据 表 timeinfo 信息 
从 以 上 运行 结果 中 可 以 看 出 ， 触 发 器 创建 成 功 后 ， 当 用 户 对 students 表 执行 DELETE 操作 时 ， 





students 数据 库 中 的 timelog 数据 表 和 timeinfo 数据 表 中 分 别 被 插入 操作 时 间 和 操作 信息 。 
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四 
二 
才 
由 


4. 
说 明 
在 MySQL 中 ， 一 个 表 在 相同 的 时 间 和 相同 的 触发 时 间 只 能 创建 一 个 触发 器 ， 如 触发 时 间 
INSERT， 和 触发 时 间 为 AFTER 的 触发 器 只 能 有 一 个 。 但 是 可 以 定义 BEFORE 的 触发 器 。 


15.2 ”查看 触发 器 


查看 触发 器 是 指 查 看 数据 库 中 已 存在 的 触发 器 的 定义 .状态 和 语法 等 信息 。 查 看 触发 器 应 用 SHOW 
TRIGGERS 语句 。 


15.2.1 SHOW TRIGGERS 


在 MySQL 中 ， 可 以 执行 SHOW TRIGGERS 语句 查看 触发 器 的 基本 信息 ， 其 基本 形式 如 下 。 

SHOW TRIGGERS; 

进入 MySQL 数据 库 ， 选 择 students 数据 库 并 查看 该 数据 库 中 存在 的 触发 器 ， 其 运行 结果 如 图 15.6 
所 示 。 


Statement 


1 Tining ! 


上 INSERT tudentinfo 
1 BEFORE ! 
上 NO_AUTO_CREATE_USER.NO_ENGINE_SUBSTITUTION ! root@localhost ! 
1 delete time_info ! DELETE ! studentinfo ! begin 
rt into timelogCsavetime> values nowC)); 


AFTIER ! NULL 


sec) 





图 15.6 查看 触发 器 


在 命令 提示 符 中 输入 SHOW TRIGGERS 语句 即 可 查看 选择 数据 库 中 的 所 有 触发 器 , 但是， 应 用 该 
查看 语句 存在 一 定 弊 端 ， 即 只 能 查询 所 有 触发 器 的 内 容 ， 并 不 能 指定 查看 某 个 触发 器 的 信息 。 这 样 一 
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来 ， 就 会 在 用 户 查找 指定 触发 器 信息 的 时 候 带 来 极 大 不 便 。 故 推荐 读者 只 在 触发 器 数量 较 少 的 情况 下 
应 用 SHOW TRIGGERS 语句 查询 触发 器 基本 信息 。 


15.2.2 ”查看 triggers 表 中 触发 器 信息 





在 MySQL 中 ， 所 有 触发 器 的 定义 都 存在 该 数据 库 的 triggers 表 中 。 读 者 可 以 通过 查询 triggers 表 
来 查看 数据 库 中 所 有 触发 器 的 详细 信息 。 查 询 语句 如 下 所 示 。 
SELECT * FROM information_schema.triggers; 


其 中 ，information_schema 是 MySQL 中 默认 存在 的 库 ， 而 information_schema 是 数据 库 中 用 于 记 
录 触 发 器 信息 的 数据 表 。 通 过 SELECT 语句 查看 触发 器 信息 。 其 运行 结果 与 图 15.6 相同 。 但 是 如 果 
户 想 要 查看 某 个 指定 触发 器 的 内 容 ， 可 以 通过 WHERE 子 句 应 用 TRIGGER 字段 作为 查询 条 件 。 其 代 
码 如 下 所 示 。 

SELECT * FROM information_schema.triggers WHERE TRIGGER_NAME= ' 触 发 器 名 称 '; 

其 中 , “触发 器 名 称 ” 这 一 参数 为 用 户 指定 要 查看 的 触发 器 名 称 ， 和 其 他 SELECT 查询 语句 相同 ， 
该 名 称 内 容 需 要 用 一 对 “"” ( 单 引 号 ) 引用 指定 的 文字 内 容 。 
NO 

如 果 数 据 库 中 存在 数量 较 多 的 触发 器 , 建议 读者 使 用 第 二 种 查看 触发 器 的 方式 。 这 样 会 在 查找 
指定 触发 器 过 程 中 避免 很 多 麻烦 。 























15.3 ”使 用 触发 器 





在 MySQL 中 ， 触 发 器 按 以 下 顺序 执行 BEFORE 触发 器 、 表 操作 、AFTER 触发 器 操作 ， 其 中 表 
操作 包括 常用 的 数据 库 操作 命令 (如 INSERT、UPDATE、DELETE) 。 

例 15.4 触发 器 与 表 操作 存在 执行 顺序 , 下 面 通过 创建 一 个 示例 向 读者 展示 三 者 的 执行 顺序 关系 。 
(实例 位 置 ， 光 盘 \TMNsINLS\1S.4) 

(1) 创建 名 称 为 before in 的 BEFORE INSERT 触发 器 ， 其 代码 如 下 。 

create trigger before_in before insert on 


studentinfo for each row 
insert into timeinfo (info) values (before'); 


(2) 创建 名 称 为 after in 的 AFTER INSERT 触发 器 ， 其 代码 如 下 。 


create trigger after_in after insert on 
studentinfo for each row 
insert into timeinfo (info) values (after ); 
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运行 步骤 (1) 、(2) 的 结果 如 图 15.7 所 示 。 


Inysql> create trigger before_in hefore insert on 

-> studentinfo for 

-> insert into tineinfo (info> values (’hefore’); 
Query OK. @ rows affected cy? 


Inysql> create trigger after_in after insert on 


nfo》 values C’after’); 





图 15.7 创建 触发 器 运行 结果 
(3) 创建 完毕 触发 器 ， 向 数据 表 studentinfo 中 插入 一 条 记录 ， 其 代码 如 下 。 
insert into studentinfo(name) values (Nowitzki); 
执行 成 功 后 ， 通 过 SELECT 语句 查看 数据 表 timeinfo 的 插入 情况 ， 其 代码 如 下 。 
select * from timeinfo; 


运行 以 上 代码 ， 其 运行 结果 如 图 15.8 所 示 。 


Ect * from tineinfo; 


2 rows in set 《0.00 sec) 





图 15.8 查看 timeinfo 表 中 触发 器 的 执行 顺序 
查询 结果 显示 BEFORE 和 AFTER 触发 器 被 激活 。 BEFORE 触发 器 首先 被 激活 , 然后 AFTER 触发 
器 再 被 激活 。 
4 
说 明 
触发 器 中 不 能 包含 START TRANSCATION、COMMIT 或 ROLLBACK 等 关键 字 , 也 不 能 包含 
CALL 语句 。 触 发 器 执行 非常 严密 ， 每 一 环 都 息息相关 ， 任 何 错误 都 可 能 导致 程序 无 法 向 下 执行 。 
已 经 更 新 过 的 数据 表 是 不 能 回 滚 的 ， 故 在 设计 过 程 中 一 定 要 注意 触发 器 的 逻辑 严密 性 。 


15.4 删除 触发 器 


在 MySQL 中 ， 既 然 可 以 创建 触发 器 ， 同 样 也 可 以 通过 命令 删除 触发 器 。 删 除 触发 器 指 删除 原来 
已 经 在 某 个 数据 库 中 创建 的 触发 器 ， 与 MySQL 中 删除 数据 库 的 命令 相似 ， 应 用 DROP 关键 字 删 除 触 
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发 器 。 其 语法 格式 如 下 。 
DROP TRIGGER 触发 器 名 称 
“触发 器 名 称 ” 参 数 为 用 户 指定 要 删除 的 触发 器 名 称 ， 如 果 指 定 某 个 特定 触发 器 名 称 ，MySQL 在 
执行 过 程 中 将 会 在 : 当前 库 中 查找 触发 器 。 
说明 
在 应 用 完 触 发 器 后 ,切记 一 定 要 将 触发 器 删除 ,否则 在 执行 某 些 数据 库 操作 时 ,会 造成 数据 的 
变化 。 





例 15.5 下 面 将 名 称 为 delete_time_info 的 触发 器 删除 ， 其 执行 代码 如 下 。【〔 实 例 位 置 ， 光 盘 
\TMNsI\MS\15.5) 


DROP TRIGGER delete_time_info; 
执行 上 述 代 码 ， 其 运行 结果 如 图 15.9 所 示 。 


ysql> use students; 
和 atabase changed 


ysql> drop trigger delete_tine_info; 
uery OK, 0 rows affected (0.11 sec) 





图 15.9 删除 触发 器 
通过 查看 触发 器 命令 来 查看 数据 库 students 中 的 触发 器 信息 ， 其 代码 如 下 。 
SHOW TRIGGERS 
查看 触发 器 信息 ， 可 以 从 图 15.10 看 出 ， 名 称 为 delete_time info 的 触发 器 已 经 被 删除 。 


ysql> show triggers; 


+ 
igger ! Event ! Table ! Statement 
Tining ! Created ! sql_node ! Definer 


1 INSERT ! studentinfo ! insert into timeinfo Cinfo> values (’hbefom 
1 NULL 1 NO_AUTO_CREATE_U -NO_ENGINE_SUBSTITUTION } root@localh 


HR udentinfo 1 i into timeinfo Cinfo> values (’after’ 
1 NULL 1 NO_AUTO_CREATE_U NO_ENGINE_SUBSTITUTION ! rootelocalh 


PR rows in set 《@.11 sec)» 








pa| 








15.10 查看 students 数据 库 中 的 触发 器 信息 
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图 15.10 的 返回 结果 显示 ， 该 数据 库 中 存在 两 个 触发 器 信息 ， 这 两 个 触发 器 是 在 15.1.2 节 中 被 
;创建 的 ， 如 果 用 户 在 db databasel15 数据 库 中 未 创建 该 触发 器 ， 则 返回 结果 会 是 一 个 “Empty set” 。 


15.5 ”小 结 


本 章 对 MySQL 数据 库 的 触发 器 的 定义 和 作用 、 创 建 触发 器 、 查 看 触发 器 、 使 用 触发 器 和 删除 触 
发 器 等 内 容 进行 了 详细 讲解 ， 创 建 触发 器 和 使 用 触发 器 是 本 章 的 重点 内 容 。 读 者 在 创建 触发 器 后 ， 一 
定 要 查看 触发 器 的 结构 。 使 用 触发 器 时 ， 触 发 器 执行 的 顺序 为 BEFORE 触发 器 、 表 操作 (INSERT、 
UPDATE 和 DELETE) 和 AFTER 触发 器 。 读 者 需要 将 本 章 的 知识 结合 实际 需要 来 设计 触发 器 。 


15.6 ”实践 与 练习 


1. 创建 一 个 由 INSERT 触发 的 触发 器 ， 实 现 当 向 department 表 中 插入 数据 时 ， 自 动向 tb_students 
表 中 插入 当前 时 间 。 答案 位 置 ， 光 盘 \TMNsN15\15.6) 

2. 删除 原 有 的 触发 器 ， 触 发 器 删除 完成 后 ， 执 行 SELECT 语句 来 查看 触发 器 是 否 还 存在 。 (答案 
位 置 : 光盘 \TMNsUN1S\1S.7) 
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( 名! 视频 讲解 ; 1S 分 钟 ) 


在 操作 My5QL 过 程 中 ， 对 于 一 般 简 单 的 业务 远 辑 或 中 小 型 程序 而 言 ， 无 须 考 
虑 应 用 MySQL 事务 。 但 在 比较 复杂 的 情况 下 ， 往 往 在 执行 基 些 教 据 操作 过 程 中 ， 
需要 通过 一 组 SQL 语句 执行 多 项 并 行业 务 远 辑 或 程序 ， 这 样 ， 就 必须 保证 所 用 命令 
执行 的 同步 性 , 使 执行 序列 中 产生 依靠 关 系 的 动作 能 够 同时 操作 成 功 或 同时 返回 初 
始 状态 。 在 此 情况 下 ， 就 需要 优先 考虑 使 用 My5QL 事务 处 理 。 


(2 


于 于 于 于 至 至 


过 阅读 本 章 ， 读 者 可 以 : 


甬 过 


了 解 MySQL 事务 的 概述 

了 解 MySQL 事务 的 创建 与 存在 周期 
掌握 MySQL 事务 的 查询 和 提交 ( 回流 ) 
掌握 MySQL 事务 行为 

掌握 MySQL 事务 的 性 能 

掌握 应 用 MySQL 伪 事 务 的 方法 
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16.1 MySQL 事务 概述 


在 MySQL 中 ， 事 务 由 单独 单元 的 一 条 或 多 条 SQL 语句 组 成 。 在 这 个 单元 中 ， 每 条 MySQL 语句 
是 相互 依赖 的 。 而 整个 单独 单元 作为 一 个 不 可 分 割 的 整体 ， 如 果 单 元 中 一 旦 某 条 SQL 语句 执行 失败 或 
产生 错误 ， 整 个 单元 将 会 回 滚 ， 所 有 受到 影响 的 数据 将 返回 到 事务 开始 以 前 的 状态 ， 如 果 单元 中 的 所 


有 SQL 语句 均 执 行 成 功 ， 则 事务 被 顺利 执行 。 


在 现实 生活 中 ， 事 务 处 理 数据 的 应 用 非常 广泛 ， 如 网 上 交易 、 银 行事 务 等 。 下 面 通 过 网 上 交易 流 


程 向 读者 展示 事务 的 概念 。 


相信 大 多 数 用 户 都 有 过 网 上 购物 的 体验 ， 即 用 户 登录 某 个 大 型 购物 网 站 ， 浏 览 该 网 站 中 所 陈列 的 
商品 信息 ， 将 喜欢 的 商品 放 入 购物 车 中 ， 选 购 完 毕 后 ， 用 户 需 要 对 选 购 的 商品 进行 在 线 支付 ， 当 用 户 
对 所 选 商 品 付款 完毕 ， 通 知 商家 发 货 ， 在 此 过 程 中 。 用 户 所 付 货款 并 未 提交 到 商户 手中 ， 当 用 户 收 到 


货物 之 后 ， 可 以 确认 收 货 ， 商 家 才 收 到 商品 货款 ， 整 个 交易 过 程 
才 算 完成 。 如 果 任 何 一 步 操作 失败 ， 则 都 会 导致 双方 陷入 尴 论 的 
境界 ， 试 想 当 用 户 选 购 商品 ， 付 款 操 作 完 成 后 ， 用 户 选择 在 发 货 
过 程 中 取消 订单 。 这 时 商家 并 没有 得 到 货款 将 取消 操作 ， 如 果 不 
应 用 事务 处 理 ， 则 用 户 在 取消 订单 操作 过 程 后 ， 商 家 仍然 继续 将 
用 户 所 订购 的 商品 发 送 给 用 户 ， 这 会 导致 一 些 不 愉快 的 争端 。 故 
在 整个 交易 过 程 中 ， 必 须 采 用 事务 来 对 网 上 交易 进行 回 滚 操作 。 
其 流程 如 图 16.1 所 示 。 

在 网 上 交易 流程 过 程 中 ， 商 家 与 用 户 的 交易 可 以 被 认为 是 一 
个 事务 处 理 过 程 ， 如 果 在 交易 流程 中 存在 一 个 环节 失败 ， 都 可 能 
导致 双方 交易 的 失败 。 如 前 面 事务 定义 所 说 ， 所 有 这 些 流程 都 应 
该 被 成 功 执行 , 在 MySQL 中 如 果 有 任何 命令 失败 , 都 会 导致 所 有 
操作 命令 被 撤销 。 系 统 返回 未 操作 前 的 状态 ， 即 回 滚 到 初始 状态 。 
添加 到 购物 车 、 在 线 付 款 、 商 家 发 货 等 构成 了 一 个 基本 的 事务 。 
整个 交易 流程 可 以 被 看 作 一 个 完整 的 单元 ， 用 于 实现 整体 事务 。 

通过 InnoDB 和 BDB 类 型 表 , MySQL 事务 能 够 完全 满足 事务 
安全 的 ACID 测试 。 但 是 并 不 是 所 有 表 类 型 都 支持 事务 ， 如 
MyISAM 类 型 表 就 不 能 支持 事务 ， 只 能 通过 伪 事 务 对 表 实 现 事务 
处 理 。 


Nem 





有 用户 通 网 商 喇 |] 类 败 








漆 加 到 购物 车 








在 线 付款 








商家 发 货 








用 户 收 货 














确认 收 货 








16.1 应 用 事务 处 理 网 上 交易 流程 


ACID 指出 每 个 事务 型 RDBMS 必须 遵守 的 4 个 属性 ， 即 原子 性 、 一 致 性 、 阪 立 性 和 持久 性 。 
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16.1.1 原子 性 


这 就 类 似 与 化 学 中 的 原子 ， 事 务 也 具备 整体 性 和 不 可 分 割 性 ， 被 认为 是 一 个 不 可 分 割 的 单元 。 假 
设 一 个 事务 由 多 种 任务 组 成 ， 其 中 的 语句 必须 同时 操作 成 功 ， 才 可 以 认为 事务 是 成 功 的 ， 否 则 将 回 深 
到 初始 状态 。 从 上 面 网 上 交易 的 例子 可 以 看 出 ， 所 有 操作 的 成 功 是 保证 交易 完成 的 前 提 条 件 ， 当 任何 
一 个 环节 出 现 问 题 时 ， 就 导致 事务 回 滨 ， 不 能 正常 完成 交易 过 程 。 即 所 有 事务 共同 进退 ， 保 证 了 事务 
的 整体 性 ， 这 就 是 事务 的 原子 性 。 

原子 的 执行 是 一 个 全 部 发 生 或 全 部 失败 的 整体 过 程 。 在 一 个 原子 操作 中 ， 如 果 事 务 中 的 任何 一 条 
语句 失败 ， 前 面 执行 的 语句 都 将 被 返回 ， 以 保证 数据 的 整体 性 不 被 破坏 。 这 在 常用 的 系统 应 用 中 ， 为 
保证 数据 的 安全 性 起 到 一 定 作用 。 


16.1.2 一 致 性 


在 MySQL 事务 处 理 过程 中 ， 无 论 事务 是 完全 成 功 或 是 在 中 途 因 某 些 环节 失败 而 导致 失败 ， 但 事 
务 使 系统 处 于 一 致 的 状态 时 ， 其 必须 保证 一 致 性 。 如 在 网 上 进行 转账 操作 的 过 程 中 ， 用 户 A 向 用 户 了 B 
的 账户 中 转 入 5 000 元 , 但 用 户 B 在 查询 转账 信息 的 时 候 , 发 现 自己 的 账户 中 只 增加 了 3 000 元 ,这样 
不 能 使 整个 事务 达到 一 致 性 。 例 如， 从 上 述 网 上 交易 的 例子 考虑 ， 当 交易 完成 后 ， 商 家 的 货物 会 减少 ， 
不 可 能 出 现 当 商家 给 用 户 发 货 后 货物 数量 不 变 ， 而 用 户 收 到 货物 不 增加 的 情况 。 

在 MySQL 中 , 一致 性 主要 由 MySQL 的 日 志 机 制 处 理 ， 它 记录 数据 库 的 所 有 变化 ， 为 事务 回复 提 
供 跟 踪 记 录 。 如 果 系 统 在 事务 处 理 中 间 发 生 错误 ，MySQL 恢复 过 程 将 使 用 这 些 日 志 发 现 事务 是 否 已 经 
完全 成 功 执行 或 需要 返回 。 一 致 性 属性 保证 数据 库 从 不 返回 一 个 未 处 理 的 事务 。 








16.1.3 ”孤立 性 


孤立 性 是 指 每 个 事务 在 自己 的 空间 发 生 ， 与 其 他 发 生 在 系统 中 的 事务 隔离 ， 而 且 事务 的 结果 只 在 
它 完 全 被 执行 时 才能 看 到 。 即 使 这 样 的 一 个 系统 中 同时 发 生 多 个 事务 ， 孤 立 性 也 可 以 保证 特定 的 事务 
在 完成 之 前 ， 其 结果 是 不 被 公布 的 。 

当 系 统 支 持 多 个 同时 存在 的 用 户 和 连接 时 ， 系 统 必须 遵守 孤立 性 原则 ， 否 则 在 执行 过 程 中 可 能 导 
致 大 量 数据 被 破坏 ， 孤 立 性 保证 每 个 事务 完整 地 在 其 各 自 的 空间 内 被 顺利 执行 ， 保 证 事务 与 事务 之 间 
不 会 相互 冲突 。 


16.1.4 ”持久 性 


在 MySQL 中 ， 即 便 是 数据 库 系 统 崩 溃 ， 一 个 被 提交 的 事务 仍然 在 坚持 。 当 一 个 事务 完成 ， 数 据 
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库 的 日 志 已 经 被 更 新 时 ， 持 久 性 即 可 发 挥 其 特有 功效 。 在 MySQL 中 ， 如 果 系 统 崩 溃 或 者 数据 存储 介 
质 被 破坏 ， 通 过 使 用 日 志 ， 系 统 能 够 恢复 在 重启 前 进行 的 最 后 一 次 成 功 更 新 ， 可 以 反映 系统 崩溃 时 处 
于 执行 过 程 的 事务 的 变化 。 

MySQL 的 持久 性 是 通过 一 条 记录 事务 过 程 中 系统 变化 的 二 进 制 事务 日 志文 件 来 实现 的 。 如 果 遇 到 
硬件 损坏 或 者 系统 的 异常 关机 ， 系 统 在 下 一 次 启动 时 ， 通 过 使 用 最 后 的 备份 和 日 志 就 可 以 恢复 丢失 
数据 。 

SO 


默认 情况 下 ，InnoDB 表 持 久 性 最 久 ，MyISAM 表 提 供 部 分 持久 。 





16.2 MySQL 事务 的 创建 与 存在 周期 


通过 上 述 对 事务 定义 的 叙述 和 事务 特性 的 讲解 ， 相 信 读 者 已 经 对 事务 有 一 个 初步 的 认识 。 下 面 的 
内 容 将 对 事务 的 存在 周期 做 详细 讲解 。 首 先 ， 向 读者 展示 在 MySQL 中 如 何 创建 事务 。 

创建 事务 的 一 般 过 程 是 : 初始 化 事务 、 创 建 事务 、 应 用 SELECT 语句 查询 数据 是 否 被 录入 和 提交 
事务 。 如 果 用 户 不 在 操作 数据 库 完 成 后 执行 事务 提交 ， 则 系统 会 默认 执行 回 滚 操 作 。 如 果 用 户 在 提交 
事务 前 选择 撤销 事务 ， 则 用 户 在 撤销 前 的 所 有 事务 将 被 取消 ， 数 据 库 系统 会 回 到 初始 状态 。 


默认 情况 下 ,在 MySQL 中 创建 的 数据 表 类 型 都 是 MyISAM, 但 是 该 类 型 的 数据 表 并 不 能 支持 
事务 。 所以, 如 果 用 户 想 让 数据 表 支 持 事务 处 理 能 力 , 必须 将 当前 操作 数据 表 的 类 型 设置 为 nnoDB 
或 BDB。 
在 创建 事务 的 过 程 中 ， 用 户 需 要 创建 一 个 InnoDB 或 BDB 类 型 的 数据 表 ， 其 基本 命令 结构 如 下 。 
CREATE TABLE table_name(field_defintions) TYPE = INNODB/BDB; 
其 中 , table_name 为 表 名 , 而 field_defintions 为 表 内 定义 的 字段 等 属性 ，TYPE 指定 数据 表 的 类 型 ， 
既 可 以 是 InnoDB 类 型 ， 也 可 以 是 BDB 类 型 。 
当 用 户 希 望 将 已 经 存在 的 表 支持 事务 处 理 ， 则 用 户 可 以 应 用 ALTER TABLE 命令 ， 指 定数 据 表 的 类 
型 即 可 实现 对 表 的 类 型 更 改 操作 ,使 原本 不 支持 事务 的 数据 表 更 改 为 支持 事务 处 理 的 类 型 ， 其 命令 如 下 。 
ALTER TABLE table_name TYPE= INNODB/BDB; 


在 用 户 更 改 完 表 的 类 型 后 ， 即 可 使 数据 表 支 持 事务 处 理 。 
BT 
| 应 用 ALTER TABLE 操作 可 能 会 导致 数据 库 中 的 数据 丢失 ， 因 此 为 了 避免 非 预期 结果 出 现 ， 
”在 使 用 ALTER TABLE 命令 之 前 ， 用 户 需要 创建 一 个 表 备份 。 
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16.2.1 初始 化 事务 


初始 化 MySQL 事务 ， 首 先 声明 初始 化 MySQL 事务 后 所 有 的 SQL 语句 为 一 个 单元 。 在 MySQL 
中 ， 应 用 START TRANSACTION 命令 来 标记 一 个 事务 的 开始 ， 初 始 化 事务 的 结构 如 下 。 

START TRANSACTION; 

另外 , 用 户 也 可 以 使 用 BEGIN 或 者 BEGIN WORK 命令 初始 化 事务 , 通常 START TRANSACTION 
命令 后 面 跟随 的 是 组 成 事务 的 SQL 语句 。 

在 命令 提示 符 中 输入 如 下 命令 。 

start transaction; 

如 果 在 用 户 输 入 以 上 代码 后 ，MySQL 数据 库 没 有 给 出 警告 提示 或 返回 错误 信息 ， 则 说 明 用 户 已 经 
事务 初始 化 成 功 ， 用 户 可 以 继续 执行 下 一 步 操作 。 








16.2.2 ”创建 事务 


例 16.1 初始 化 事务 成 功 后 ， 可 以 创建 事务 。 这 里 以 向 名 称 为 connection 的 数据 表 中 插入 一 条 记 
录 为 例 ， 讲 解 事务 的 创建 。 首 先 打开 数据 库 ， 选 定 某 个 数据 库 ， 然 后 初始 化 事务 ， 最 后 创建 事务 ， 向 
指定 的 数据 表 中 添加 记录 ， 其 代码 如 下 。 《实例 位 置 : 光盘 \TMNsN16\16.1) 

mysql -uroot —proot 

use db_database16 ; 

start transaction; 


insert into connection(email,cellphone,QQ,sid) 
values('barrystephen@126.com',13456000000,187034000,3); 


其 运行 结果 如 图 16.2 所 示 。 


ffecte 3 ‘0.00 sec》 


into connectionCem. ellphone-99.sid> 
Ca .13456808080.187034800.3》; 





图 16.2 创建 事务 
16.2.3 应 用 SELECT 语句 查询 数据 是 否 被 正确 录入 


事务 创建 成 功 后 ， 建 议 读 者 通过 SELECT 语句 查询 数据 是 否 被 正确 录入 。 
例 16.2 在 事务 初始 化 成 功 后 ， 用 户 创建 事务 ,继续 在 命令 提示 符 中 输入 如 下 指令 。 (实例 位 置 : 
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光盘 \TMNsN\16\16.2) 
SELECT * FROM connection WHERE «sid=3; 


其 运行 结果 如 图 16.3 所 示 。 


1 187834969 ! 3 1 





图 16.3 查询 数据 是 否 被 正确 录入 


售 明 

在 用 户 插入 新 表 为 InhnoDB 类 型 或 更 改 原来 表 类 型 为 InnoDB 时 ， 如 果 在 输入 命令 提示 后 ， 
MySQL 提示 “The 'InnoDB' feature is disabled; you need 'InnoDB' to have it working” 警 告 ， 则 说 明 
InnoDB 表 类 型 并 没有 被 开启 ， 用 户 需 要 找到 MySQL 文件 目录 下 的 my.ini 文 件 ， 定 位 skip innodb 
选项 位 置 ， 将 原来 的 skip innodb 改 为 “#skip innodb” 后 保存 该 文件 ， 重 新 启动 MySQL 服务 器 ， 
即 可 令 数 据 库 支持 InnoDB 类 型 表 。 


16.2.4 ”提交 事务 


在 用 户 没有 提交 事务 之 前 ， 当 其 他 用 户 连接 MySQL 服务 器 时 ， 应 用 SELECT 语句 查询 结果 ， 则 

会 时 不 没有 拓 交 的 事务 。 当 且 仅 当 用 户 成 功 提交 事务 后 ， 其 他 用 户 才 可 能 通过 SELECT 语 ee 
务 结果 4。 由 事务 的 特性 可 知 ， 事 务 具 有 孤立 性 ， 当 事务 处 在 处 理 过 程 中 ， 其 实 MySQL 并 未 将 结果 
入 磁盘 中 ， 这 样 一 来 ， 这 些 正在 处 理 的 事务 相对 其 他 用 户 是 不 可 见 的。 一 旦 数据 被 正确 插入 ， 
以 使 用 COMMIT 命令 提交 事务 。 提 交 事 务 的 命令 结构 如 下 。 

COMMIT 


- 旦 当前 执行 事务 的 用 户 提交 当前 事务 ， 则 其 他 用 户 就 可 以 通过 会 话 查询 结果 。 
16.2.5 ”撤销 事务 (事务 回 滚 ) 


撤销 事务 ， 又 被 称 作 事务 回 滚 ， 即 事务 被 用 户 开启 、 用 户 输入 的 SQL 语句 被 执行 后 ， 如 果 用 户 想 要 
撤销 刚才 的 数据 库 操作 , 可 使 用 ROLLBACK 命 令 撤 销 数据 库 中 的 所 有 变化 -ROLLBACK 命令 结构 如 下 。 

ROLLBACK 

输入 回 滚 操作 后 ， 如 何 判断 是 否 执行 回 滚 操 作 了 呢 ? 可 以 通过 SELECT 语句 查看 16.2.2 节 中 插入 
的 数据 是 否 存在 ， 其 运行 结果 如 图 16.4 所 示 。 











259 


MySQL 从 入 门 到 精通 


nysql> select x from connection where sid=3; 
Enpty set 《0.808 sec》> 





ysql> 





图 16.4 执行 回 滚 操作 后 应 用 SELECT 查询 











$b 注 总 

如 果 执 行 一 个 回 滚 操作 ， 则 在 输入 START TRANSACTIONA 命令 后 的 所 有 SQL 语句 都 将 执 
行 回 滚 操作 。 故 在 执行 事务 回 滚 前 ， 用 户 需要 慎重 选择 执行 回 滚 操作 。 如 果 用 户 开 局 事务 后 ， 没 有 
提交 事务 ， 则 事务 默认 为 自动 回 滚 状态 ， 即 不 保存 用 户 之 前 的 任何 操作 。 


Ww 
说 明 
在 现实 应 用 中 ,事务 撤 销 即 事务 回 滚 有 着 很 重要 的 意义 。 例如， 用 户 A 和 用 户 B 采 用 银行 转账 方 
式 交 易 ， 用 户 A 将 个 人 账户 的 部 分 存款 转移 到 用 户 B 的 个 人 账户 过 程 中 ， 若 银行 的 数据 库 系 统 突然 发 
生 错 误 或 异常 ， 则 交易 事务 提交 失败 ， 系 统 执行 回 滚 操作 ， 恢 复 到 交易 的 初始 状态 。 这 样 ， 采 用 事务 
回 滚 可 以 避免 因 特 殊 情况 而 导致 事务 提交 失败 ， 从 而 导致 不 必要 的 损失 。 


16.2.6 事务 的 存在 周期 


事务 的 周期 由 用 户 在 命令 提示 符 中 输入 START TRANSACTION 指令 开始 ， 直 至 用 户 输 入 
COMMIT 结束 ， 图 16.5 展示 了 一 个 简单 事务 存在 周期 流程 图 。 





\ 


( 初始 化 数据 库 状 态 太一 









































{ 新 的 数据 库 状 态 广 


人 


图 16.5 事务 的 存在 周期 
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/ 

说 明 

事务 不 支持 谋 套 功能 ， 当 用 户 在 未 结束 第 一 个 事务 又 重新 打开 一 个 事务 时 , 则 前 一 个 事务 会 自 
动 提交 。 在 MySQL 中 很 多 Re COMMIT 命令 。 


16.3 ”MySQL 事务 行为 





在 MySQL 中 ,存在 两 个 可 以 控制 行为 的 变量 , 它们 分 别 是 AUTOCOMMIT 变量 和 TRANSACTION 
ISOLACTIONLEVEL 变量 。 


16.3.1 自动 提交 


在 MySQL 中 ， 如 果 不 更 改 其 自动 提交 变量 ， 则 系统 会 自动 向 数据 库 提 交 结 果 ， 用 户 在 执行 数据 
库 操作 过 程 中 ， 不 需要 使 用 START TRANSACTION 语句 开始 事务 ， 应 用 COMMIT 或 者 ROLLBACK 
提交 事务 或 执行 回 滚 操作 。 如 果 用 户 希望 通过 控制 MySQL 自动 提交 参数 ， 可 以 更 改 提交 模式 ， 这 
更 改过 程 是 通过 设置 AUTOCOMMIT 变量 来 实现 的 。 

下 面 通过 一 个 示例 向 读者 展示 如 何 关闭 自动 提交 参数 ， 在 命令 提示 符 中 输入 以 下 命令 。 

SET AUTOCOMMIT=0; 





关闭 自动 提交 功能 。 只 有 当 用 户 输入 COMMIT 命令 后 ，MySQL 才 将 数据 表 中 的 资料 提交 到 数据 
库 中 ， 如 果 不 提交 事务 ， 而 终止 MySQL 会 话 ， 数 据 库 将 会 自动 执行 回 滚 操作 。 

例 16.3 在 关闭 自动 提交 命令 后 ， 查 询 数据 表 中 数据 ， 并 且 向 数据 表 中 添加 一 条 记录 ， 其 命令 如 
下 。“〈 实 例 位 置 ; 光盘 \TMsIN16\16.3) 


select * from timeinfo; 
insert into timeinfo(info) values('test autocommit); 


以 上 代码 的 运行 结果 如 图 16.6 所 示 。 


autocommit=8; 
8 rows affected (8.88 sec» 


ect x* from tineinfo; 


valuesC’test autocomnit’); 





16.6 取消 自动 提交 操作 


261 


MySQL 从 入 门 到 精通 

















由 于 用 户 已 经 关闭 自动 提交 功能 ， 所 以 图 16.6 中 所 示 的 添加 操作 由 于 没有 执行 事务 的 提交 操作 ， 
导致 数据 没有 成 功 添加 。 再 次 查询 数据 表 中 的 数据 ， 运 行 结果 如 图 16.7 所 示 。 

结果 表明 ， 之 前 插入 的 数据 并 未 插入 到 数据 库 中 ， 另 外 ， 可 以 通过 查看 @@AUTOCOMMIT 变量 
来 查看 当前 自动 提交 状态 ， 查 看 此 变量 同样 应 用 SELECT 语句 ， 其 运行 结果 如 图 16.8 所 示 。 





lect BRautocommit; 


一 + 


8.00 sec》 





图 16.7 应 用 SELECT 语句 查询 自动 提交 关闭 后 的 数据 图 16.8 查看 自动 提交 变量 
16.3.2 事务 的 孤立 级 


事务 具有 独立 的 空间 ， 在 MySQL 服务 器 中 ， 用 户 通过 不 同 的 会 话 执行 不 同 的 事务 ， 在 多 用 户 环 
境 中 ， 许 多 RDBMS 会 话 在 任意 指定 时 刻 都 是 活动 的 。 为 了 使 这 些 事务 互 不 影响 ， 保 证 数据 库 性 能 不 
受到 影响 ， 采 用 事务 的 孤立 级 是 十 分 有 必要 的 。 

孤立 级 在 整个 事务 中 起 到 了 很 重要 的 作用 ， 如 果 没有 这 些 事务 的 孤立 性 ， 不 同 的 SELECT 语句 将 
会 在 同一 事务 的 环境 中 检索 到 不 同 的 结果 ， 这 将 导致 数据 的 不 一 致 性 。 给 不 同 的 用 户 造 成 困扰 ， 这 样 

-来 ， 用 户 就 不 能 将 查询 的 结果 集 作为 计算 基础 。 所 以 孤立 性 强制 保持 每 个 事务 的 独立 性 ， 以 此 来 保 
证 事务 中 看 到 一 致 的 数据 。 

基于 ANSIISO SQL 规范 ，MySQL 提供 以 下 4 种 孤立 级 。 

(1) SERIALIZABLE (序列 化 ): 顾名思义 ， 以 序列 的 形式 对 事务 进行 处 理 ， 该 孤立 级 的 特点 是 
只 有 当 事 务 提 交 后 ， 用 户 才能 从 数据 库 中 查看 数据 的 变化 。 该 孤立 级 运行 会 影响 MySQL 的 性 能 。 因 
为 需要 占用 大 量 资 源 ， 以 保证 使 大 量 事务 在 任意 时 间 不 被 用 户 看 到 。 

(2) REPEATABLE READ (可 重读 ) : 对 于 应 用 程序 的 安全 性 作出 部 分 妥协 ， 以 提高 其 性 能 。 事 
务 在 该 孤立 级 上 不 会 被 看 成 一 个 序列 ， 不 过 当前 在 执行 事务 的 过 程 中 ， 用 户 仍然 看 不 到 事务 的 过 程 。 
直到 事务 提交 为 止 ， 用 户 才 能 够 看 到 事务 的 变化 结果 。 

(3) READ COMMITTED (提交 后 读 ) : 提交 后 读 孤 立 级 的 安全 性 比重 复读 安全 性 要 低 。 在 这 一 
级 的 事务 ， 用 户 可 以 看 到 其 他 事务 添加 的 新 记录 。 在 事务 处 理 时 ， 如 果 存 在 其 他 用 户 同时 对 事务 的 相 
应 表 进 行 修 改 ， 那 么 在 同一 事务 中 不 同时 间 内 ， 应 用 SELECT 语句 可 能 返回 不 同 的 结果 集 。 

(4) READ UNCOMMITTED (未 提交 读 ) : 该 孤立 级 提供 事务 之 间 的 最 小 程度 间隔 ， 该 孤立 级 容 
易 产 生 虚 幻 读 操作 。 其 他 用 户 可 以 在 该 孤立 级 上 看 到 未 提交 的 事务 。 























16.3.3 ”修改 事务 的 孤立 级 





在 MySQL 中 ,可 以 使 用 TRANSACTION ISOLATION LEVEL 变量 修改 事务 孤立 级 , 其 中 , MySQL 
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的 默认 孤立 级 为 REPEATABLE READ (可 重读 ) ,用户 可 以 使 用 SELECT 命令 获取 当前 事务 孤立 级 
变量 的 值 ， 其 命令 如 下 。 

SELECT @@itx_isolation ; 

例 16.4 如 果 用 户 希 望 修改 事务 的 孤立 级 ,可 以 通过 SET 命 令 设 置 其 不 同 值 来 修改 事务 的 孤立 级 ， 
操作 命令 如 图 16.9 所 示 。《〔 实 例 位 置 ， 光 盘 \TMNsN\16\16.4) 





> set global transaction isolation level 


Committedj 
ss _ affected 〈《B.BB sec》 





图 16.9 设置 事务 孤立 级 


YX 
说 明 
如 果 用 户 想 修改 事务 的 孤立 级 ， 必 须 首先 获取 SUPER 优先 权 ， 以 便 用 户 可 以 顺利 执行 修改 


操作 。 


16.4 事务 的 性 能 


从 16.3 节 中 可 以 看 出 ， 应 用 不 同 孤 立 级 的 事务 可 能 会 对 系统 造成 一 系列 影响 ， 采 用 不 同 孤 立 级 处 
理事 务 ， 可 能 会 对 系统 稳定 性 和 安全 性 等 诸多 因素 造成 影响 。 另 外 ， 有 些 数 据 库 操作 中 ， 不 需要 应 用 
事务 处 理 ， 则 用 户 在 选择 数据 表 类 型 时 ， 需 要 选择 合适 的 数据 表 类 型 。 所 以 ， 在 选择 表 类 型 时 ， 应 该 
考虑 数据 表 具 有 完善 的 功能 ， 且 高 效 执行 的 前 提 下 ， 也 不 会 对 系统 增加 额外 的 负担 。 








16.4.1 应 用 小 事务 


应 用 小 事务 的 意义 在 于 : 保证 每 个 事务 不 会 在 执行 前 等 待 很 长 时 间 ， 从 而 避免 各 个 事务 因为 互相 
等 待 而 导致 系统 性 能 的 大 幅度 下 降 。 用 户 在 应 用 少数 大 事务 的 时 候 ， 可 能 无 法 看 出 因 事务 间 互 相等 待 
而 导致 系统 性 能 下 降 ， 但 是 当 系统 中 存在 处 理 量 很 大 的 数据 库 或 多 种 复杂 事务 的 时 候 ， 用 户 就 可 以 明 
显 感觉 到 事务 因 长 时 间 等 待 ， 而 导致 系统 性 能 下 降 。 所 以 ， 应 用 小 事务 可 以 保证 系统 的 性 能 ， 其 可 以 
快速 变化 或 退出 ， 这 样 ， 其 他 在 队列 中 准备 就 绪 的 事务 就 不 会 受到 明显 影响 。 


16.4.2 ”选择 合适 的 孤立 级 
因为 事务 的 性 能 与 其 对 服务 器 产生 的 负载 成 反比 ， 即 当 事 务 孤立 级 越 高 ， 其 性 能 越 低 ， 但 是 其 安 


全 性 也 越 高 。 其 性 能 和 孤立 级 关系 如 图 16.10 所 示 。 所 以 只 有 选择 适当 的 孤立 级 ， 才 能 有 效 地 提高 
MySQL 系统 性 能 和 应 用 性 。 
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可 重读 十 


提交 后 读 十 


未 提交 读 十 











图 16.10 ”事务 孤立 级 性 能 关系 


从 图 16.10 可 以 看 出 , 虽然 随 着 孤立 级 的 增高 稳定 性 也 会 随 之 改变 , 但 是 并 不 代表 孤立 级 的 稳定 性 
越 高 ， 也 不 能 表明 其 灵活 性 越 高 ， 故 用 户 在 选择 孤立 级 的 时 候 ， 需 要 根据 自身 实际 情况 选择 适合 应 用 
的 孤立 级 ， 切 勿 生 搬 硬 套 。 


16.4.3” 死 锁 的 概念 与 避免 


死 锁 ， 即 当 两 个 或 者 多 个 处 于 不 同 序列 的 用 户 打算 同时 更 新 某 相 同 的 数据 库 时 ， 因 互相 等 待 对方 
释放 权限 而 导致 双方 一 直 处 于 等 待 状态 。 在 实际 应 用 中 ， 两 个 不 同 序列 的 客户 打算 同时 对 数据 执行 操 
作 ， 极 有 可 能 产生 死 锁 。 更 具体 地 讲 ， 当 两 个 事务 相互 等 待 操作 对 方 释放 所 持 有 的 资源 ， 而 导致 两 个 
事务 都 无 法 操作 对 方 持 有 的 资源 ， 这 样 无 限期 的 等 待 被 称 作 死 锁 。 

不 过 ，MySQL 的 InnoDB 表 处 理 程序 具有 检查 死 锁 这 一 功能 ， 如 果 该 处 理 程序 发 现 用 户 在 操作 过 
程 中 产生 死 锁 ， 该 处 理 程序 立刻 通过 撤销 操作 来 撤销 其 中 一 个 事务 ， 以 便 使 死 锁 消失 。 这 样 就 可 以 使 
另 一 个 事务 获取 对 方 所 占有 的 资源 而 执行 逻辑 操作 。 


16.5 MySQL 伪 事 务 


在 MySQL 中 ，InnoDB 和 BDB 类 型 表 可 以 支持 事务 处 理 ， 但 是 MySQL 中 MyISAM 类 型 表 并 不 
能 支持 事务 处 理 ， 对 于 某 些 应 用 该 类 型 的 数据 表 ， 用 户 可 以 选择 应 用 表 锁 定 来 替代 事务 。 这 种 引用 表 
锁定 来 蔡 代 事务 的 事件 被 称 作伪 事务 。 使 用 表 锁 来 锁定 表 的 操作 ， 可 以 加 强 非 事 务 表 在 执行 过 程 中 的 
安全 性 和 稳定 性 。 
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16.5.1 用 表 锁 定 代替 事务 








在 MySQL 的 MyISAM 类 型 数据 表 中 ， 并 不 支持 COMMIT (提交 ) 和 ROLLBACK ( 回 滚 ) 命令 。 
当 用 户 对 数据 库 执 行 插入 、 删 除 、 更 新 等 操作 时 ， 这 些 变化 的 数据 都 被 立刻 保存 在 磁盘 中 。 这 样 ， 在 
多 用 户 环 境 中 ， 会 导致 诸多 问题 ， 为 了 避免 同一 时 间 有 多 个 用 户 对 数据 库 中 指定 表 作 。 可 以 应 
用 表 锁 定 来 避免 在 用 户 操作 数据 表 过 程 中 受到 干扰 。 当 且 仅 当 该 用 户 释放 表 的 操作 锁定 后 ， 其 他 用 户 
才 可 以 访问 这 些 修改 后 的 数据 表 。 

设置 表 锁 定 代替 事务 基本 步骤 如 下 。 

(1) 为 指定 数据 表 添 加 锁定 ， 其 语法 如 下 。 

LOCK TABLES table_name lock_type,…- 

其 中 , table name 为 被 锁定 的 表 名 , lock _type 为 锁定 类 型 ， 该 类 型 包括 以 读 方式 (READ ) 锁定 表 ; 
以 写 方式 (WRITE) 锁定 表 。 

(2) 用 户 执行 数据 表 的 操作 ， 可 以 添加 、 删 除 或 者 更 改 部 分 数据 。 

(3) 用 户 完成 对 锁定 数据 表 的 操作 后 ， 需 要 对 该 表 进 行 解锁 操作 ， 释 放 该 表 的 锁定 状态 ， 其 语法 
如 下 。 

UNLOCK TABLES 

下 面 通过 实例 向 读者 展示 如 何以 读 方 式 锁定 数据 表 和 以 写 方式 锁定 数据 表 。 

1. 以 读 方 式 锁定 数据 表 

以 读 方式 锁定 数据 表 ， 该 方式 是 设置 锁定 用 户 的 其 他 方式 操作 ， 如 删除 、 插 入 、 更 新 都 不 被 允许 ， 
直至 用 户 进行 作 。 

例 16.5 演示 以 读 方 式 锁定 表 后 ， 向 该 表 中 插入 数据 ， 以 及 解锁 后 再 插入 数据 的 情况 。〔 实 例 位 
置 : 光盘 \TMNsN\N16\16.5) 

首先 ， 以 锁定 数据 表 studentinfo 为 例 ， 在 命令 提示 符 下 输入 如 下 代码 。 

lock table studentinfo read; 

然后 ， 应 用 SELECT 语句 查看 表 中 的 信息 ， 其 运行 结果 如 图 16.11 所 示 。 














































图 16.11 查看 以 读 方式 锁定 的 studentinfo 表 
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下 面 尝 试 向 该 数据 表 中 插入 一 条 数据 ， 其 运行 结果 如 图 16.12 所 示 。 


x.tel)values 


not locked with LOCK TABLES 





图 16.12 向 以 读 方 式 锁定 的 表 中 插入 数据 


从 上 述 结果 可 以 看 出 ， 当 用 户 试 图 向 数据 库 插入 数据 时 ， 将 会 返回 失败 信息 。 当 用 户 将 锁定 的 表 
解锁 后 ， 再 次 执行 插入 操作 ， 其 运行 结果 如 图 16.13 所 示 。 








图 16.13 向 解锁 后 的 数据 表 中 添加 数据 


锁定 被 释放 后 ， 用 户 可 以 对 数据 库 执行 添 加 、 删 除 、 更 新 等 操作 。 
/ 
说 明 
其 中 ,lock type 参数 中 ,用 户 指定 数据 表 以 读 方式 (READ ) 锁 定数 据 表 的 变 体 为 READ LOCAL 
锁定 ， 其 与 READ 锁定 的 不 同 点 是 ， 该 参数 所 指定 的 用 户 会 话 可 以 执行 INSERT 操作 。 它 是 为 了 
使 用 MySQL dump 工具 而 创建 的 一 种 变 体 形式 。 


2. 以 写 方式 锁定 数据 表 

与 读 方式 锁定 表 类 似 ， 表 的 写 锁 定 是 设置 用 户 可 以 修改 数据 表 中 的 数据 ， 但 是 除 自己 以 外 其 他 会 
话 中 的 用 户 不 能 进行 任何 读 操 作 。 在 命令 提示 符 中 输入 如 下 命令 。 

lock table studentinfo write; 


例 16.6 ”因为 该 表 为 写 锁定 ， 用 户 可 以 对 数据 库 的 数据 执行 修改 、 添 加 、 删 除 等 操作 。 在 该 命令 
提示 符 中 应 用 SELECT 语句 查询 该 锁定 表 , 其 运行 结果 如 图 16.14 所 示 。( 实 例 位 置 :光盘 \TMsN16\16.6) 





Ex from studentinfos 


<B-BB 











16.14 查询 应 用 写 操作 锁定 的 数据 表 studentinfo 
从 图 16.14 中 ， 读 者 可 以 看 到 ， 当 前 用 户 应 用 SELECT 语句 仍然 可 以 查询 该 表 的 数据 ， 并 没有 限 











266 


第 16 章 事务 的 应 用 


制 用 户 对 数据 表 的 读 操作 。 这 是 因为 ， 以 写 方式 锁定 数据 表 并 不 能 限制 当前 锁定 用 户 的 查询 操作 ， 下 
面 打开 一 个 新 用 户 会 话 ， 即 保持 如 图 16.14 所 示 窗 口 不 被 关闭 ， 重 新 打开 一 个 新 的 MySQL 连接 ， 并 执 
行 上 述 过 程 。 其 运行 结果 如 图 16.15 所 示 。 





























图 16.15 打开 新 会 话 查询 被 锁定 的 数据 表 
在 新 打开 的 命令 提示 界面 ， 读 者 可 以 看 到 ， 应 用 SELECT 语句 执行 查询 操作 ， 并 没有 结果 显示 ， 


这 是 因为 之 前 该 表 以 写 方式 锁定 。 故 当 操 作用 户 释放 该 数据 表 锁 定 后 ， 其 他 用 户 才 可 以 通过 SELECT 
语句 查看 之 前 被 锁定 的 数据 表 。 在 命令 提示 符 中 输入 如 下 代码 。 
UNLOCK TABLES; 


这 时 ， 在 第 二 次 打开 的 命令 提示 符 中 ， 即 可 出 现 如 图 16.14 所 示 的 结果 。 当 数据 表 被 释放 锁定 后 ， 
其 他 访问 数据 库 的 用 户 即 可 查看 数据 表 的 内 容 。 


VT 


使 用 UNLOCK TABLE 命令 后 ， 将 会 释放 所 有 当前 处 于 锁定 状态 的 数据 表 。 


16.5.2 ”应 用 表 锁 实现 伪 事 务 


相信 读者 通过 上 面 的 学 习 ， 已 经 了 解 什么 是 表 锁 。 下 面 的 示例 通过 使 用 表 锁 对 MyISAM 表 进 行 锁 

定 操作 ， 以 此 过 程 来 代替 事务 型 表 InnoDB， 即 应 用 表 锁 来 实现 伪 事 务 。 实 现 伪 事 务 的 一 般 步 又 如 下 。 
(1) 对 数据 库 中 的 数据 表 进 行 锁定 操作 ， 可 以 对 多 个 表 做 不 同 的 方式 锁定 ， 其 代码 格式 如 下 。 
LOCK TABLE table_name1 lock_type1, table_name2 lock_type2…- 


(2) 执行 数据 库 操作 ， 向 锁定 的 数据 表 中 执行 添加 、 删 除 、 修 改 操 等 操作 。 

如 前 面 提 到 的 INSERT、UPDATE、DELETE 等 操作 。 用 户 可 以 对 锁定 的 数据 表 执 行 上 述 操作 ， 在 
执行 过 程 中 ， 该 伪 事 务 所 产生 的 结果 是 不 会 被 其 他 用 户 更 改 的 。 

(3) 释放 锁定 的 数据 表 ， 以 便 让 正在 队列 中 等 待 查看 或 操作 的 其 他 用 户 可 以 浏览 数据 表 中 的 数据 
或 对 操作 表 执 行 各 种 数据 的 操作 。 

如 果 存 在 其 他 会 话 要 求 访问 已 锁定 的 多 个 表格 , 则 该 会 话 必须 被 迫 等 待 当前 锁定 用 户 释放 锁定 表 
才 允 许 其 他 会 话 访问 该 数据 表 ， 表 锁定 使 不 同 会 话 执行 的 数据 库 操作 彼此 独立 。 应 用 数据 表 锁 定 方式 
可 以 使 不 支持 事务 类 型 的 表 实 现 伪 事务 。 
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16.6 小 结 


本 章 对 MySQL 中 对 事务 的 创建 、 提 交 、 撤 销 ， 以 及 存在 周期 进行 了 详细 讲解 ， 并 通过 举例 说 明 ， 
帮助 读者 更 好 地 理解 所 学 知识 的 用 法 。 在 阅读 本 章 时 ， 读 者 应 该 重点 掌握 事务 如 何 自 动 提交 ， 并 且 修 
改 事务 的 孤立 级 。 同 时 读者 应 该 了 解 MySQL 事务 和 性 能 ， 以 及 MySQL 伪 事 务 。 


16.7 实践 与 练习 


1. 在 PHP 中 使 用 事务 处 理 技术 实现 银行 的 安全 转账 。 (答案 位 置 ; 光盘 \TMNsN16\16.7) 
2. 在 Java 中 实现 模拟 银行 转账 系统 ， 通 过 事务 保证 转账 业务 的 顺利 进行 。 (答案 位 置 ; 光盘 
\TMNsIN16\16.8) 
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s/s 


事 件 


在 系统 管理 或 者 数据 库 管理 中 ， 经 常 要 周期 性 地 执行 某 一 个 命令 或 者 SQL 话 
句 。My5QL 在 5.1 以 后 推出 了 事件 调度 器 (Event Scheduler) ， 可 以 很 方便 地 实现 
My5QL 数据 库 的 计划 任务 ， 定 期 运行 指定 命令 ， 使 用 起 来 非常 简单 和 方便 。 

通过 阅读 本 章 ， 读 者 可 以 : 


豆 豆 于 于 至 


了 解 查 看 事件 是 否 开 启 的 方法 

党 所 开启 事件 的 方法 

党 握 使 用 创建 事件 语句 创建 事件 的 方法 

党 所 使 用 修改 事件 语句 修改 事件 ， 以 及 临时 关闭 事件 的 方法 
了 解 删除 事件 的 方法 


MySQL 从 入 门 到 精通 


17.1 事件 概述 


在 MySQL 5.1 中 新 增 了 一 个 特色 功能 事件 调度 器 (Event Scheduler)， 简 称 事件 。 它 可 以 作为 定时 
任务 调度 器 ， 取代 部 分 原来 只 能 用 操作 系统 的 计划 任务 才能 执行 的 工作 。 另 外 ， 更 值得 一 提 的 是 ， 
MySQL 的 事件 可 每 秒 钟 执行 一 个 任务 ， 这 在 一 些 对 实时 性 要 求 较 高 的 环境 下 是 非常 实用 的 。 

事件 调试 器 是 定时 触发 执行 的 ， 从 这 个 角度 上 看 也 可 以 称 作 是 “临时 触发 器 ”。 但 是 它 与 触发 器 又 
有 所 区 别 ， 触 发 器 只 针对 某 个 表 产 生 的 事件 执行 一 些 语句 ， 而 事件 调度 器 则 是 在 某 一 段 〈 间 隔 ) 时 间 
执行 一 些 语句 。 




















17.1.1 ”查看 事件 是 否 开启 


事件 由 一 个 特定 的 线程 来 管理 。 启 用 事件 调度 器 后 ， 拥 有 SUPER 权限 的 账户 执行 SHOW 
PROCESSLIST 命令 就 可 以 看 到 这 个 线程 了 

例 17.1 查看 事件 是 否 开启 ， 具体 代码 如 下 。 《实例 位 置 : 光盘 \TMsN17\17.1) 

SHOW VARIABLES LIKE 'event_scheduler ; 


SELECT @@event_scheduler; 
SHOW PROCESSLIST 





运行 以 上 代码 的 结果 如 图 17.1 所 示 。 

















17.1 查看 事件 是 否 开 启 
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从 图 17.1 中 可 以 看 出 事件 没有 开启 , 因为 参数 event_scheduler 的 值 为 OFF, 并 且 在 PROCESSLIST 
中 查看 不 到 event_scheduler 的 信息 ; 而 如 果 参 数 event_scheduler 的 值 为 ON, 或 者 在 PROCESSLIST 中 
显示 了 event_scheduler 的 信息 ， 那 么 就 说 明 事件 已 经 开启 。 





17.1.2 ”开启 事件 


通过 设 定 全 局 变量 event_scheduler 的 值 即 可 动态 地 控制 事件 调度 器 是 否 启用 。 开 启 MySQL 的 事件 
调度 器 ， 可 以 通过 下 面 两 种 方式 实现 。 

1. 通过 设置 全 局 参数 修改 

在 MySQL 的 命令 行 窗口 中 ， 使 用 SET GLOBAL 命令 可 以 开启 或 关闭 事件 。 将 event_scheduler 参 
数 的 值 设置 为 ON， 则 开启 事件 ， 如 果 设 置 为 OFF， 则 关闭 事件 。 例 如 ， 要 开启 事件 可 以 在 命令 行 窗 
口中 输入 下 面 的 命令 。 

SET GLOBAL event_scheduler = ON; 

例 17.2 ”开启 事件 并 查看 事件 是 否 已 经 开启 ， 有 具体 代码 如 下 。 《实例 位 置 ; 光盘 \TMNsIN17\17.2) 


SET GLOBAL event_scheduler = ON; 
SHOW VARIABLES LIKE 'event_scheduler ; 


运行 以 上 代码 的 结果 如 图 17.2 所 示 。 


1> SET GLOBAL event_scheduler = ON; 
Ok, @ rows affected (0.01 sec) 








OW UARIABLES LIKE ’event_scheduler’; 


ent_scheduler 上 ON 


1 row in set (8@.88 sec》 





nysql> 。 


图 17.2 开启 事件 并 查看 事件 是 否 已 经 开启 
从 图 17.2 中 ， 可 以 看 出 event_scheduler 的 值 为 ON， 则 表示 事件 已 经 开启 。 
和 ot 总 
如 果 想 要 始终 开启 事件 ， 那 么 在 使 用 SET GLOBAL 开启 事件 后 ， 还 需要 在 my.ini/my.cnf 中 添 
加 event_scheduler=on。 因 为 如 果 没 有 添加 ，MySQL 重启 事件 又 会 回 到 原来 的 状态 。 


2. 更 改 配置 文件 


在 MySQL 的 配置 文件 myini (Windows 系统 ) /my.cnf (Linux 系统 ) 中 ， 找 到 [mysqld]， 然 后 在 
下 面 添 加 以 下 代码 开启 事件 。 
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event_scheduler=ON 

在 配置 文件 中 添加 代码 并 保存 文件 后 ， 还 需要 重新 启动 MySQL 服务 器 才能 生效 。 通 过 该 方法 开 
启事 件 ， 重 启 MySQL 服务 器 后 ， 不 恢复 为 系统 默认 的 未 开启 状态 。 例 如， 此 时 重新 连接 MySQL 服务 
器 ,然后 使 用 下 面 的 命令 查看 事件 是 否 开启 时 ,得 到 的 结果 将 是 参数 event_scheduler 的 值 为 ON， 表示 
已 经 开启 ， 如 图 17.3 所 示 。 


MySQL 5.6 Command Line co 


TT TE 








”to clear the current input statenent. 








图 17.3 查看 事件 是 否 开启 


17.2 创建 事件 


在 MySQL 5.1 以 上 版 本 中 ， 可 以 通过 CREATE EVENT 语句 来 创建 事件 ， 其 语句 格式 如 下 。 


CREATE 
[DEFINER = { user | CURRENT_USER 
EVENT [IF NOT EXISTS] event_name 
ON SCHEDULE schedule 
[ON COMPLETION [NOT] PRESERVE] 
[ENABLE | DISABLE | DISABLE ON SLAVE] 
[COMMENT 'comment] 
DO event_body; 


从 上 面 的 语法 中 ， 可 以 看 出 CREATE EVENT 语句 由 多 个 子 句 组 成 ， 各 子 句 的 详细 说 明 如 表 17.1 所 示 。 


表 17.1 CREATE EVENT 语句 的 子 句 





子 名 说 了 明 
DEFINER 可 选 ， 用 于 定义 事件 执行 时 检查 权限 的 用 户 
IF NOT EXISTS 可 选 ， 用 于 判断 要 创建 的 事件 是 否 存在 








必 选 ， 用 于 指定 事件 名 ，event_name 的 最 大 长 度 为 64 个 字符 ， 如 果 未 指定 
event_ name， 则 默认 为 当前 的 MySQL 用 户 名 〈 不 区 分 大 小 写 ) 
ON SCHEDULE schedule 必 选 ， 用 于 定义 执行 的 时 间 和 时 间 间 隔 


EVENT event name 
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续 表 
子 名 说 明 
ON COMPLETION [NOT] 可 选 , 用 于 定义 事件 是 否 循环 执行 , 即 是 一 次 执行 还 是 永久 执行 , 默认 为 一 次 执行 ， 
PRESERVE 即 NOT PRESERVE 








可 选 ， 用 于 指定 事件 的 一 种 属性 。 其 中 ， 关 键 字 ENABLE 表示 该 事件 是 活动 的 ， 
也 就 是 调度 器 检查 事件 是 否 必须 调用 ; 关键 字 DISABLE 表示 该 事件 是 关闭 的 ， 也 


BNABLBIDISABLE! 就 是 事件 的 声明 存储 到 目录 中 ， 但 是 调度 器 不 会 检查 它 是 否 应 该 调用 ， 关 键 字 











NE DISABLE ON SLAVE 表示 事件 在 从 机 中 是 关闭 的 。 如 果 不 指定 这 三 个 选项 中 的 任 
何 一 个 ， 则 在 一 个 事件 创建 之 后 ， 它 立即 变 为 活动 的 

COMMENT 'comment 可 选 ， 用 于 定义 事件 的 注释 
必 选 ， 用 于 指定 事件 启动 时 所 要 执行 的 代码 。 可 以 是 任何 有 效 的 SQL 语句 、 存 储 

DO event body 过 程 或 者 一 个 计划 执行 的 事件 。 如 果 包 含 多 条 语句 ， 可 以 使 用 BEGIN-…END 复合 
结构 


在 ON SCHEDULE 子 句 中 ， 参 数 schedule 的 值 为 一 个 As 子 句 ， 用 于 指定 事件 在 某 个 时 刻 发 生 ， 
其 语法 格式 如 下 。 
AT timestamp [+ INTERVAL interval] … 
| EVERY interval 


[STARTS timestamp [+ INTERVAL interval] …] 
[ENDS timestamp [+ INTERVAL interval] …] 


参数 说 明 如 下 。 

(1) timestamp: 表示 一 个 具体 的 时 间 点 ， 后 面 加 上 一 个 时 间 间 隔 ， 表 示 在 这 个 时 间 间 隔 后 事件 
发 生 。 

(2) EVERY 子 句 : 用 于 表示 事件 在 指定 时 间 区 间 内 每 隔 多 长 时 间 发 生 一 次 ， 其 中 STARTS 子 句 
用 于 指定 开始 时 间 ; ENDS 子 句 用 于 指定 结束 时 间 。 

(3) interval: 表示 一 个 从 现在 开始 的 时 间 ， 其 值 由 一 个 数值 和 单位 构成 。 例 如 ,使 用 “4 WEEK” 
表示 4 周 ; 使 用 “'1:10' HOUR_MINUTE” 表 示 1 小 时 10 分 钟 。 问 隔 的 距离 用 DATE_ADDO 函 数 来 
支配 。 

interval 参数 值 的 语法 格式 如 下 。 

quantity {YEAR | QUARTER | MONTH | DAY | HOUR | MINUTE | 

WEEK | SECOND | YEAR_MONTH | DAY_HOUR | 
DAY_MINUTE IDAY_SECOND | HOUR_MINUTE | 
HOUR_SECOND | MINUTE_SECOND} 

例 17.3 在 数据 库 db_database17 中 创建 一 个 名 称 为 e_test 的 事件 ， 用 于 每 隔 5 秒 钟 向 数据 表 
tb_eventtest 中 插入 一 条 数据 。〔 实 例 位 置 ， 光盘 \TMNsNM7\17.3) 

(1) 打开 数据 库 db_database17， 代 码 如 下 。 

use db_database17; 


(2) 创建 名 称 为 e_test 的 事件 , 用 于 每 隔 5 秒 钟 向 数据 表 tb_eventtest 中 插入 一 条 数据 , 代码 如 下 。 
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CREATE EVENT IF NOT EXISTS e_test ON SCHEDULE EVERY 5 SECOND 
ON COMPLETION PRESERVE 
DO INSERT INTO tb_eventtest(user,createtime) VALUES(root,NOW()); 
(3) 创建 事件 后 ， 编 写 以 下 查看 数据 表 tb_eventtest 中 数据 的 代码 。 
select * from tb_eventtest; 
执行 结果 如 图 17.4 所 示 ， 从 该 图 中 可 以 看 出 ， 每 隔 5 秒 钟 插入 一 条 数据 ， 这 说 明 事件 已 经 创建 
成 功 。 





= 了 nn 
国 wsQt 56 Command Line Client 


E EUENT IF NOT E t ON SCHEDULE EUERY 5 SECOND 
LETION 了 
etine》 VALUESC’ root? -NOW 


ot 《8.88 sec) 








图 17.4 创建 事件 e_test 
例 17.4 创建 一 个 事件 ， 实 现 每 个 月 的 第 一 天 凌晨 1 点 统计 一 次 已 经 注册 的 会 员 人 数 ， 并 插入 到 统 
计 表 中 。《〈 实 例 位 置 ; 光盘 \TMNsIN17\17.4) 
(1) 创建 名 称 为 p_total 的 存储 过 程 ， 用 于 统计 已 经 注册 的 会 员 人 数 ， 并 插入 到 统计 表 tb total 中 ， 
具体 代码 如 下 。 
DELIMITER// 


create procedure p_total() 
begin 


DECLARE n_total INT default 0; 
select COUNT(") into n_total FROM db_database17 .tb_user; 
INSERT INTO tb_total (userNumber,createtime) values(n_total,NOW()); 


end 
ll 


(2) 创建 名 称 为 e_autoTotal 的 事件 ， 用 于 在 每 个 月 的 第 一 天 凌晨 1 点 调用 步骤 (1) 中 创建 的 存储 
过 程 p_ total， 代 码 如 下 。 


CREATE EVENT IF NOT EXISTS e_autoTotal 
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ON SCHEDULE EVERY 1 MONTH 

STARTS DATE_ADD(DATE_ADD(DATE_SUB(CURDATE(),INTERVAL DAY(CURDATE())-1 DAY), 
INTERVAL 1 MONTH),INTERVAL 1 HOUR) 

ON COMPLETION PRESERVE ENABLE 

DO CALL p_total(); 


创建 存储 过 程 的 执行 结果 如 图 17.5 所 示 ; 创建 事件 的 执行 结果 如 图 17.6 所 示 。 


nysql> use db_database17; 
Database changed 
nysql> DELIMITER // 
1> create procedure p_totalC) 
-> begin 
-> 
-> DECLARE n_total INT default @; 
> select COUNTCx> into n_total FROM db_database11.th.s 
-> INSERT INTO th_total CuserNunber,createtine> valuesCn_total,NOWC); 
-> 
end 
-》 vv 
Query 0K。 0 rows affected (0.03 





图 17.5 ”创建 存储 过 程 p_total 


DATE_ADDCDATE_ADDCDAT! ¢CURDATEC >, INTERUVAL URDATECY> -1 DAY>, INTERUAL 





图 17.6 ”创建 事件 e_autoTotal 


17.3 修改 事件 


在 MySQL 5.1 及 以 后 版 本 中 ， 事 件 被 创建 之 后 ， 还 可 以 使 用 ALTER EVENT 语句 修改 其 定义 和 相 
关 属 性 ， 其 语法 格式 如 下 。 


ALTER 
[DEFINER = { user | CURRENT_USERj 
EVENT event_name 
[ON SCHEDULE schedule] 
[ON COMPLETION [NOT] PRESERVE] 
[RENAME TO new_event_name] 
[ENABLE | DISABLE | DISABLE ON SLAVE] 
[COMMENT 'comment] 
[DO event_body] 


ALTER EVENT 语句 的 使 用 语法 与 CREATE EVENT 语句 基本 相同 ,这 里 不 再 歼 述 。 另 外 ，ALTER 
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EVENT 语句 还 有 一 个 用 法 就 是 让 一 个 事件 关闭 或 再 次 让 其 活动 。 不 过 需要 注意 的 是 ， 一 个 事件 最 后 
次 被 调用 后 ， 它 是 无 法 被 修改 的 ， 因 为 此 时 它 已 经 不 存在 了 。 
例 17.5 ”修改 例 17.3 中 创建 的 事件 ， 让 其 每 隔 30 秒 向 数据 表 tb_eventtest 中 插入 一 条 数据 。〈 实 
例 位 置 ， 光盘 \TMNsM7\17.5) 
(1) 在 MySQL 的 命令 行 窗口 中 ,编写 修改 事件 的 代码 ， 有 具体 代码 如 下 。 
ALTER EVENT e_test ON SCHEDULE EVERY 30 SECOND 
ON COMPLETION PRESERVE 
DO INSERT INTO tb_eventtest(user,createtime) VALUES(root,NOW()); 
(2) 编写 查询 数据 表 中 数据 的 代码 ， 有 具体 代码 如 下 。 
SELECT * FROM tb_eventtest; 


执行 结果 如 图 17.7 所 示 。 


UENT e ON SCHEDULE EVERY 3@ SECOND 
LETION PRESERVE 


迅 














Frv-cFreatetime》 UALUESC’root’ .NOWC); 





图 17.7 修改 名 称 为 e_test 的 事件 


/ 
说 明 
从 图 17.7 的 查询 结果 中 可 以 看 出 ， 在 修改 事件 后 ， 表 tb_eventtest 中 的 数据 由 原来 的 每 5 秒 插 
入 一 条 ， 改 变 为 每 30 秒 插入 一 条 。 


应 用 ALTER EVENT 语句 ， 还 可 以 临时 关闭 一 个 已 经 创建 的 事件 ， 下 面 将 举例 进行 说 明 。 
例 17.6 ”临时 关闭 例 17.3 中 创建 的 事件 e_test。《〈 实 例 位 置 ; 光盘 \TMsIN17\17.6) 

(1) 在 MySQL 的 命令 行 窗口 中 ， 编 写 临 时 关闭 事件 e_test 的 代码 ， 具 体 代码 如 下 。 
ALTER EVENT e_test DISABLE; 

(2) 编写 查询 数据 表 中 数据 的 代码 ， 具 体 代码 如 下 。 
SELECT * FROM tb_eventtest; 





为 了 查看 事件 是 否 关 闭 , 可 以 执行 两 次 (每 次 间隔 1 分 钟 ) 步骤 (2) 中 的 代码 , 执行 结果 如 图 17.8 
所 示 。 
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th_eventtest; 


4 rows in set ‘(0.80 sec) 





图 17.8 临时 关闭 名 称 为 e_test 的 事件 


/ 
说 明 
从 图 17.8 的 查询 结果 中 可 以 看 出 ， 临 时 关闭 事件 后 ， 将 不 再 继续 向 数据 表 tb_eventtest 中 插入 
数据 。 


17.4 删除 事件 


在 MySQL 5.1 及 以 后 版 本 中 ， 删 除 已 经 创建 的 事件 可 以 使 用 DROP EVENT 语句 来 实现 。DROP 
EVENT 语句 的 语法 格式 如 下 。 

DROP EVENT [IF EXISTS] event_name 

例 17.7 删除 例 17.3 中 创建 的 事件 e_test， 代 码 如 下 。 “实例 位 置 ， 光盘 \ITMNsINI7\17.7) 


use db_database17 
DROP EVENT IF EXISTS e_test'; 


执行 结果 如 图 17.9 所 示 。 





mwsal》 use db_database17; 
pa changed 


ROP EVENT IF EXISTS e_t 
. 0 rows affected (8.80 sec) 





图 17.9 删除 名 称 为 e_test 的 事件 
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17.3* 水 结 


本 章 首 先 介绍 了 什么 是 事件 、 如 何 查看 事件 是 否 开 启 ， 以 及 如 何 开启 事件 ， 然 后 介绍 了 如 何 创 建 、 
修改 和 删除 事件 。 其 中 ， 如 何 开启 、 创 建 、 修 改 和 删除 事件 是 本 章 的 重点 ， 需 要 读者 认真 学 习 并 做 到 
融会 贯通 ， 为 以 后 的 工作 和 学 习 打 下 良好 的 基础 。 





17.6 ”实践 与 练习 


1. 创建 事件 ， 实 现 每 隔 一 个 月 清空 一 次 新 闻 信息 表 。 (答案 位 置 ， 光盘 \TMNsN17\17.8) 
2. 在 数据 库 db_database17 中 创建 一 个 名 称 为 e_test 的 事件 ， 用 于 每 隔 1 分 钟 向 数据 表 tb_eventtest 
中 插入 一 条 数据 。 (答案 位 置 ; 光盘 \TMsIN17\17.9) 
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# Os 


备份 与 恢复 


(名! 视 频 讲 解 : 3 分 钟 ) 


为 了 保证 数据 的 安全 ， 需 要 定期 对 数据 进行 备份 。 备 份 的 方式 有 很 多 种 ， 效 果 
也 不 一 样 。 如 果 数 据 库 中 的 数据 出 现 了 错误 ， 就 需要 使 用 备份 好 的 数据 进行 数据 还 
原 。 这 样 可 以 将 损失 降 至 最 低 。 而 且 , 可 能 还 会 涉及 数据 库 之 间 的 数据 导入 与 导出 。 
本 章 将 介绍 备份 和 还 原 的 方法 ， 对 My5QL 数据 库 的 数据 安全 等 内 容 进行 讲解 。 

通过 阅读 本 章 ， 读 者 可 以 : 

让 ”掌握 数据 备份 的 使 用 方法 

Mm 掌握 数据 恢复 的 方法 

MW 掌握 数据 库 迁 移 的 方法 

掌握 导出 和 导入 文本 文件 的 方法 
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18.1 数据 备份 


备份 数据 是 数据 库 管理 最 常用 的 操作 。 为 了 保证 数据 库 中 数据 的 安全 ， 数 据 管理 员 需 要 定期 进行 
数据 备份 。 一 旦 数据 库 遭 到 破坏 ， 即 通过 备份 的 文件 来 还 原 数据 库 。 

可 能 造成 数据 损坏 的 原因 很 多 ， 大 体 上 可 以 归纳 为 以 下 几 个 方面 。 

(1) 存储 介质 故障 : 保存 数据 库 文件 的 磁盘 设备 损坏 ， 同 时 又 没有 对 数据 库 备份 ， 从 而 导致 数据 
彻底 丢失 。 

(2) 服务 器 彻底 瘫痪 : 数据 库 服 务 器 彻底 瘫痪 ， 系 统 需 要 重建 。 

(3) 用 户 的 误 操作 : 在 删除 数据 库 时 ， 不 小 心 删除 了 某 些 重要 数据 ， 或 者 是 整个 数据 库 。 

(4) 黑客 破坏 :系统 唱 到 黑客 的 恶意 攻击 ， 数 据 或 者 数据 表 被 删除 。 

因此 ， 数 据 备份 是 很 重要 的 工作 。 本 节 将 介绍 数据 备份 的 方法 。 


18.1.1 使 用 mysqldump 命令 备份 


mysqldump 命令 可 以 将 数据 库 中 的 数据 备份 成 一 个 文本 文件 。 表 的 结构 和 表 中 的 数据 将 存储 在 生 
成 的 文本 文件 中 。 本 节 将 介绍 mysqldump 命令 的 工作 原理 和 使 用 方法 。 

mysqldump 命令 的 工作 原理 很 简单 。 它 先 查 出 需要 备份 的 表 的 结构 ， 再 在 文本 文件 中 生成 一 条 
CREATE 语句 。 然 后 ， 将 表 中 的 所 有 记录 转换 成 一 条 INSERT 语句 。 这 些 CREATE 语句 和 JINSERT 语 
句 都 是 还 原 时 使 用 的 。 还 原 数据 时 就 可 以 使 用 其 中 的 CREATE 语句 来 创建 表 。 使 用 其 中 的 INSERT 语 
句 来 还 原 数据 。 

在 使 用 mysqldump 命令 进行 数据 备份 时 ， 经 常 分 为 以 下 3 种 形式 。 

(1) 备份 一 个 数据 库 。 

(2) 备份 多 个 数据 库 。 

(3) 备份 所 有 数据 库 。 

下 面 将 分 别 介绍 如 何 实现 这 3 种 形式 的 数据 备份 。 


1. 备份 一 个 数据 库 
使 用 mysqldump 命令 备份 一 个 数据 库 的 基本 语法 如 下 。 
mysqldump -u username -p dbname table1 table2 ...>BackupName.sql 


其 中 ，dbname 参数 表示 数据 库 的 名 称 ; tablel 和 table2 参数 表示 表 的 名 称 ， 没 有 该 参数 时 将 备份 
整个 数据 库 ; BackupName.sql 参数 表示 备份 文件 的 名 称 ， 文 件 名 前 面 可 以 加 上 一 个 绝对 路 径 。 通 常 将 
数据 库 备 份 成 一 个 后 级 名 为 .sql 的 文件 。 
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4 
| mysqldump 命令 备份 的 文件 并 非 一 定 要 求 后 级 名 为 .sql， 备 份 成 其 他 格式 的 文件 也 是 可 以 的 ， 
“例如 ， 后 组 名 为 .txt 的 文件 。 但 是 ， 通 常情 况 下 是 备份 成 后 级 名 为 .sql 的 文件 。 因 为 ， 后 级 名 为 .sql 

的 文件 给 人 第 一 感觉 就 是 与 数据 库 有 关 的 文件 。 





例 18.1 下 面 使 用 root 用 户 备份 test 数据 库 下 的 student 表 ， 命 令 如 下 。《〔 实 例 位 置 ， 光 盘 
\TMNsI\18\18.1) 

mysqldump -u root -p test student >D:\ student.sql 

在 DOS 命令 窗口 中 执行 上 面 的 命令 时 ， 将 提示 输入 连接 数据 库 的 密码 ， 输 入 密码 后 将 完成 数据 备 
份 ， 这 时 可 以 在 D:\ 中 找到 student.sql 文件 。student.sql 文件 中 的 部 分 内 容 如 图 18.1 所 示 。 
stuclent.sql - MR-N..XT.master (sa (51))| 了 


1 -- MySQL dump 10.13 Distrib 5.5.12, for 
日 Win32 (x86) 








aN 
! 
! 





-- Server version 5.5.12 


| /*!140101 SET BOLD CHARACTER SET CLIENT=B@ 

| CHARACTER SET CLIENT */; 

/*!40101 SET BOLD CHARACTER SET RESULTS=BB 
| CHARACTER SET RESULTS */; 

/*!140101 SET BOLD_COLLATION CONNECTION=B8 

| COLLATION_CONNECTION */; 

/*!40101 SET NANMES utf8 */; 

11| /*!40103 SET BOLD_TINE ZONE=8@TINE ZONE */; 
12| /*!40103 SET TINE ZONE='+00:00' */; 
/*!40014 SET BOLD_UNIQUE CHECKS=B@ 


图 18.1 备份 一 个 数据 库 
文件 开头 记录 了 MySQL 的 版 本 、 备 份 的 主机 名 和 数据 库 名 。 文 件 中 ， 以 “--” 开 头 的 都 是 SQL 
的 注释 。 以 “/*! 40101” 等 形式 开头 的 内 容 是 只 有 MySQL 版 本 大 于 或 等 于 指定 的 版 本 4.1.1 才 执 行 的 
语句 。 下 面 的 “/*! 40103”“/*! 40014” 也 是 这 个 作用 。 














上 面 student.sql 文件 中 没有 创建 数据 库 的 语句 ， 因 此 ，student.sq] 文件 中 的 所 有 表 和 记录 必须 
还 原 到 一 个 已 经 存在 的 数据 库 中 。 还原 数据 时 ，CREATE TABLE 语句 会 在 数据 库 中 创建 表 ， 然 后 


执行 INSERT 语句 向 表 中 插入 记录 。 


2. 备份 多 个 数据 库 

mysqldump 命令 备份 多 个 数据 库 的 语法 如 下 。 

mysqldump -u username —p --databases dbname1 dbname2 >BackupName.sql 
这 里 要 加 上 databases 这 个 选项 ， 然 后 后 面 跟 多 个 数据 库 的 名 称 。 
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例 18.2 下 面 使 用 root 用 户 备份 test 数据 库 和 mysql 数据 库 ， 命 令 如 下 。 (实例 位 置 ， 光盘 
\TMNsI\18\18.2) 











mysqldump —u root -p --databases test mysql >D:\backup.sql 

在 DOS 命令 窗口 中 执行 上 面 的 命令 时 ， 将 提示 输入 连接 数据 库 的 密码 ， 输 入 密码 后 将 完成 数据 备 
份 ， 这 时 可 以 在 D:\ 下 面 看 到 名 为 backup.sql 的 文件 ， 如 图 18.2 所 示 。 这 个 文件 中 存储 着 这 两 个 数据 库 
的 所 有 信息 。 





名 称 
出 币 用 职 忻 | 具 
但 6 襄 列 子 
琶 喜 炉 
企业 ERP 管 理 系统 
网 站 


BB backup.sql 

围 MingRilogLDF 
GMingRi3.mdf 
国 Student.mdf 





18.2 备份 多 个 数据 库 

3. 备份 所 有 数据 库 

mysqldump 命令 备份 所 有 数据 库 的 语法 如 下 。 

mysqldump -u username —p --all -databases >BackupName.sql 

使 用 --all -databases 选项 就 可 以 备份 所 有 数据 库 了 。 

例 18.3 下 面 使 用 root 用 户 备份 所 有 数据 库 。 命 令 如 下 。 (实例 位 置 : 光盘 \TMNsN18\18.3) 

mysqldump -u root -p --all -databases >D:\all.sql 

在 DOS 命令 窗口 中 执行 上 面 的 命令 时 ,将 提示 输入 连接 数据 库 的 密码 , 输入 密码 后 将 完成 数据 备份 ， 
这 时 可 以 在 D\ 下 面 看 到 名 为 all.sql 的 文件 ， 如 图 18.3 所 示 。 这 个 文件 存储 着 所 有 数据 库 的 所 有 信息 。 


名 称 
网 站 


国 allsql 





国 backup.sql 
国 MingRi logLDF 


加 MingRi3.mdf 
国 Student.mdf 
国 student.sql 

国 Student logJdf 





图 18.3 备份 所 有 数据 库 
18.1.2 ”直接 复制 整个 数据 库 目录 


MySQL 有 一 种 最 简单 的 备份 方法 ， 就 是 将 MySQL 中 的 数据 库 文件 直接 复制 出 来 。 这 种 方法 最 简 
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单 ， 速 度 也 最 快 。 使 用 这 种 方法 时 ， 最 好 将 服务 器 先 停止 。 这 样 ， 可 以 保证 在 复制 期 间 数据 库 中 的 数 
据 不 会 发 生变 化 。 如 果 在 复制 数据 库 的 过 程 中 还 有 数据 写 入， 就 会 造成 数据 不 一 致 。 


wel 
| 为 了 保证 所 备份 数据 的 完整 性 ， 在 停止 MySQL 服务 器 之 前 ,需要 先 执行 FLUSH TABLES 语 
句 将 所 有 数据 写 入 到 数据 文件 的 文本 文件 里 。 


这 种 方法 虽然 简单 快捷 ， 但 不 是 最 好 的 备份 方法 。 因 为 ， 实 际 情况 可 能 不 允许 停止 MySQL 服务 
器 。 而 且 ， 这 种 方法 对 InnoDB 存储 引擎 的 表 不 适用 。 对 于 MyISAM 存储 引擎 的 表 ， 这 样 备份 和 还 原 
很 方便 。 但 是 还 原 时 最 好 是 相同 版 本 的 MySQL 数据 库 ， 否 则 可 能 会 存储 文件 类 型 不 同 的 情况 。 


SC 
在 MySQL 的 版 本 号 中 ， 第 一 个 数字 表示 主 版 本 号 。 主 版 本 号 相同 的 MySQL 数据 库 的 文件 类 
型 会 相同 。 例 如 ，MySQL 5.1.39 和 MYSQL 5.1.40 这 两 个 版 本 的 主 版 本 号 都 是 5， 那么 这 两 个 数据 
库 的 数据 文件 拥有 相同 的 文件 格式 。 


18.1.3 ”使 用 mysqlhotcopy 工具 快速 备份 


如 果 备 份 时 不 能 停止 MySQL 服务 器 ， 可 以 采用 mysqlhotcopy 工具 。mysqlhotcopy 工具 的 备份 方 
式 比 mysqldump 命令 快 。 下 面 介绍 mysqlhotcopy 工具 的 工作 原理 和 使 用 方法 。 

mysqlhotcopy 工具 是 一 个 Perl 脚本 , 主要 在 Linux 操作 系统 下 使 用 。mysqlhotcopy 工具 使 用 LOCK 
TABLES、FLUSH TABLES 和 cp 来 进行 快速 备份 。 其 工作 原理 是 ， 先 将 需要 备份 的 数据 库 加 上 一 个 读 
操作 锁 ， 然 后 , 用 FLUSH TABLES 将 内 存 中 的 数据 写 回 到 硬盘 上 的 数据 库 中 ， 最 后 ， 把 需要 备份 的 数 
据 库 文件 复制 到 目标 目录 。 使 用 mysqlhotcopy 的 命令 如 下 。 

[root@localhost ~]#mysqlhotcopy[option] dbname1 dbname2.…backupDir 


其 中 ，dbnamel 等 表示 需要 备份 的 数据 库 的 名 称 ，backupDir 参数 指出 备份 到 哪个 文件 夹 下 。 这 个 
命令 的 含义 就 是 将 dbnamel、dbname2 等 数据 库 备份 到 backDir 目录 下 。mysqlhotcopy 工具 有 一 些 常用 
的 选项 ， 这 些 选项 的 介绍 如 下 。 

(1) --help: 用 来 查看 mysqlhotcopy 的 帮助 。 

(2) --allowold: 如 果 备 份 目录 下 存在 相同 的 备份 文件 ， 将 旧 的 备份 文件 名 加 上 _old。 

(3) --keepold: 如 果 备 份 目录 下 存在 相同 的 备份 文件 ， 不 删除 旧 的 备份 文件 ， 而 是 将 旧 文件 更 名 。 

(4) --flushlog: 本 次 备份 之 后 ， 将 对 数据 库 的 更 新 记录 到 日 志 中 。 

(5) --noindices: 只 备份 数据 文件 ， 不 备份 索引 文件 。 

(6) --user- 用 户 名 : 用 来 指定 用 户 名 ， 可 以 用 a 代替 。 

(7) --password= 密 码 : 用 来 指定 密码 ， 可 以 用 -p 代替 。 使 用 -p 时 ， 密 码 与 -p 紧 挨 着 。 或 者 只 使 用 
-p， 然 后 用 交换 的 方式 输入 密码 。 这 与 登录 数据 库 时 的 情况 是 一 样 的 。 

(8) --por 人 = 端口 号 : 用 来 指定 访问 端口 ， 可 以 用 -P 代替。 
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(9) --socket=socket 文件 : 用 来 指定 socket 文件 ， 可 以 用 -S 代替 。 


mysqlhotcopy 工具 不 是 MySQL 自 带 的 ， 需 要 安装 Perl 的 数据 接口 包 ，Perl 的 数据 库 接 口 包 可 
以 在 MySQL 官方 网 站 下 载 ， 网 址 是 http://dev.mysql.com/downloads/dbi.html。mysqlhotcopy 工具 的 
工作 原理 是 将 数据 库 文件 复制 到 目标 目录 。 因 此 mysqlhotcopy 工具 只 能 备份 MyISAM 类 型 的 表 ， 
不 能 用 来 备份 InoDB 类 型 的 表 。 


18.2 数据 恢复 


管理 员 的 非法 操作 和 计算 机 的 故障 都 会 破坏 数据 库 文件 。 当 数据 库 遇 到 这 些 意外 时 ， 可 以 通过 备 
份 文件 将 数据 库 还 原 到 备份 时 的 状态 。 这 样 可 以 将 损失 降低 到 最 小 。 本 节 将 介绍 数据 还 原 的 方法 。 


18.2.1 使 用 mysql 命令 还 原 


通常 使 用 mysqldump 命令 将 数据 库 的 数据 备份 成 一 个 文本 文件 。 通 常 这 个 文件 的 后 缀 名 是 .sql。 需 
要 还 原 时 ， 可 以 使 用 mysql 命令 来 还 原 备 份 的 数据 。 

备份 文件 中 通常 包含 CREATE 语句 和 INSERT 语句 。mysql 命令 可 以 执行 备份 文件 中 的 CREATE 
语句 和 INSERT 语句 。 通 过 CREATE 语句 来 创建 数据 库 和 表 。 通 过 INSERT 语句 来 插入 备份 的 数据 。 
mysql 命令 的 基本 语法 如 下 。 

mysql -uroot -p [dbname] <backup.sql 


其 中 ，dbname 参数 表示 数据 库 名 称 。 该 参数 是 可 选 参数 ， 可 以 指定 数据 库 名 ， 也 可 以 不 指定 。 指 
定数 据 库 名 时 ， 表 示 还 原 该 数据 库 下 的 表 。 不 指定 数据 库 名 时 ， 表 示 还 原 特定 的 一 个 数据 库 。 备 份 文 
件 中 有 创建 数据 库 的 语句 。 

例 18.4 下 面 使 用 root 用 户 备份 所 用 数据 库 ， 命 令 如 下 。 实例 位 置 : 光盘 \TMNsN18\18.4) 

mysql -u root -p <D:\all.sql 


在 DOS 命令 窗口 中 执行 上 面 的 命令 时 ， 将 提示 输入 连接 数据 库 的 密码 ， 输 入 密码 后 将 完成 数据 还 
原 。 这 时 ，MySQL 数据 库 就 已 经 还 原 了 all.sql 文件 中 的 所 有 数据 库 。 























如 果 使 用 --all-databases 参数 备份 了 所 有 的 数据 库 ， 那 么 还 原 时 不 需要 指定 数据 库 。 因 为 ， 其 
;对 应 的 .sql 文件 包含 CREATE DATABASE 语句 ， 可 以 通过 该 语句 创建 数据 库 .创建 数据 库 之 后 ， 
;可 以 执行 sql 文件 中 的 USE 语句 选择 数据 库 ， 然 后 在 数据 库 中 创建 表 并 且 插 入 记录 。 
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18.2.2 ”直接 复制 到 数据 库 目 录 


之 前 介绍 过 一 种 直接 复制 数据 的 备份 方法 。 通 过 这 种 方式 备份 的 数据 ， 可 以 直接 复制 到 MySQL 
的 数据 库 目 录 下 。 通 过 这 种 方式 还 原 时 ， 必 须 保 证 两 个 MySQL 数据 库 的 主 版 本 号 是 相同 的 。 而 且 ， 
这 种 方式 对 MyISAM 类 型 的 表 比 较 有 效 。 对 于 InnoDB 类 型 的 表 则 不 可 用 。 因 为 InnoDB 表 的 表 空间 
不 能 直接 复制 。 

在 Windows 操作 系统 下 ，MySQL 的 数据 库 目 录 通 常 存放 在 下 面 3 个 路 径 的 其 中 之 一 ， 分 别 是 
C:\mysql\date、 C:\Documents and Settings\All Users\Application Data\MySQL\AMYyYSQL Server5.1\data 或 者 
Ci\Program Files\MySQL Server 5.1\data。 在 Linux 操作 系统 下 ， 数 据 库 目录 通常 在 /varlib/mysql/、 
/usrlocal/mysql/data 或 者 /usr/local/mysql/var 这 3 个 目录 下 。 上 述 位 置 只 是 数据 库 目 录 最 常用 的 位 置 。 
具体 位 置 根据 读者 安装 时 设置 的 位 置 而 定 。 

使 用 mysqlhotcopy 命令 备份 的 数据 也 是 通过 这 种 方式 来 还 原 的 。 在 Linux 操作 系统 下 ， 复 制 到 数 
据 库 目录 后 ， 一 定 要 将 数据 库 的 用 户 和 组 变 成 mysql， 命 令 如 下 。 

chown -R mysql.mysql dataDir 

其 中 , 两 个 mysql 分 别 表示 组 和 用 户 ; -R 参数 可 以 改变 文件 夹 下 的 所 有 子 文件 的 用 户 和 组 ; dataDir 
参数 表示 数据 库 目录 。 








Linux 操作 系统 下 的 权限 设置 非常 严格 。 通 常情 况 下 ，MySQL 数据 库 只 有 root 用 户 和 mysql 
用 户 组 下 的 mysql 用 户 可 以 访问 。 因 此 ， 将 数据 库 目 录 复 制 到 指定 文件 夹 后 ， 一 定 要 使 用 chown 
命令 将 文件 夹 的 用 户 组 变 为 mysql， 将 用 户 变 为 mysql。 


18.3 数据库 迁 移 


数据 库 迁 移 就 是 指 将 数据 库 从 一 个 系统 移动 到 另 一 个 系统 上 。 数 据 库 迁移 的 原因 是 多 种 多 样 的 。 
可 能 是 因为 升级 了 计算 机 ， 或 者 是 部 署 开 发 的 管理 系统 ， 或 者 是 升级 了 MySQL 数据 库 ， 甚 至 是 换 
其 他 的 数据 库 。 根 据 上 述 情况 ， 可 以 将 数据 库 迁 移 大 致 分 为 3 类 ， 分 别 是 在 相同 版 本 的 MySQL 数据 
库 之 间 迁 移 、 迁 移 到 其 他 版 本 的 MySQL 数据 库 中 和 迁移 到 其 他 类 型 的 数据 库 中 。 本 节 将 介绍 数据 库 
迁移 的 方法 。 

















18.3.1 相同 版 本 的 MySQL 数据 库 之 间 的 迁移 
相同 版 本 的 MySQL 数据 库 之 问 的 迁移 就 是 在 主 版 本 号 相同 的 MySQL 数据 库 之 间 进 行 数据 库 移 
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动 。 这 种 迁移 的 方式 最 容易 实现 。 本 节 将 介绍 这 方面 的 内 容 。 

相同 版 本 的 MySQL 数据 库 之 问 进行 数据 库 迁 移 的 原因 很 多 。 通 常 的 原因 是 换 了 新 的 机 器 ， 或 者 
是 装 了 新 的 操作 系统 。 还 有 一 种 常见 的 原因 就 是 将 开发 的 管理 系统 部 署 到 工作 机 器 上 。 因 为 迁移 前 后 
MySQL 数据 库 的 主 本 版 号 相同 ， 所 以 可 以 通过 复制 数据 库 目录 来 实现 数据 库 迁 移 。 但 是 ， 只 有 数据 库 
表 都 是 MYISAM 类 型 的 才能 使 用 这 种 方式 。 

最 常用 和 最 安全 的 方式 是 使 用 mysqldump 命令 来 备份 数据 库 。 然 后 使 用 mysql 命令 将 备份 文件 还 
原 到 新 的 MySQL 数据 库 中 。 这 里 可 以 将 备份 和 迁移 同时 进行 。 假 设 从 一 个 名 为 hostl 的 机 器 中 备份 出 
所 有 数据 库 ， 然 后 ， 将 这 些 数据 库 迁 移 到 名 为 host2 的 机 器 上 。 命 令 如 下 。 


mysqldump -h name1 -u root -password=password1 -all-databases | 
mysql -h host2 -u root -password=password2 


其 中 ，“|” 符 号 表示 管道 ， 其 作用 是 将 mysqldump 备份 的 文件 送 给 mysql 命令 ，“-password= 
password1” 是 namel 主机 上 root 用 户 的 密码 ， 同 理 ，password2 是 name2 主机 上 的 root 用 户 的 密码 。 
通过 这 种 方式 可 以 直接 实现 迁移 。 





18.3.2 不 同 数据 库 之 间 的 迁移 


不 同 数据 库 之 间 迁 移 是 指 从 其 他 类 型 的 数据 库 迁 移 到 MySQL 数据 库 , 或 者 从 MySQL 数据 库 迁 移 
到 其 他 类 型 的 数据 库 。 例 如 ， 某 个 网 站 原来 使 用 Oracle 数据 库 ， 因 为 运营 成 本 太 高 等 诸多 原因 ， 和 希望 
改 用 MySQL 数据 库 ; 或 者 ， 某 个 管理 系统 原来 使 用 MySQL 数据 库 ， 因 为 某 种 特殊 性 能 的 要 求 ， 希 望 
改 用 Oracle 数据 库 。 这 样 的 不 同 数据 库 之 问 的 迁移 也 经 常会 发 生 ， 但 是 这 种 迁移 没有 普通 适用 的 解决 
方法 。 

MySQL 以 外 的 数据 库 也 有 类 似 mysqldump 这 样 的 备份 工具 ， 可 以 将 数据 库 中 的 文件 备份 成 .sql 文 
件 或 普通 文件 。 但 是 ， 因 为 不 同 数据 库 厂 商 没 有 完全 按照 SQL 标准 来 设计 数据 库 ， 这 就 造成 了 不 同 数 
据 库 使 用 的 SQL 语句 的 差异 。 例 如 ， 微 软 的 SQL Server 软件 使 用 的 是 T-SQL。T-SQL 中 包含 非 标准 
的 SQL 语句 。 这 就 造成 了 SQL Server 和 MySQL 的 SQL 语句 不 能 兼容 。 

除了 SQL 语句 存在 不 兼容 的 情况 下 ， 不 同 的 数据 库 之 间 的 数据 类 型 也 有 差异 。 数 据 类 型 的 差异 也 
造成 了 迁移 的 困难 。 例如 ，SQL Server 数据 库 中 有 ntext、 Image 等 数据 类 型 , 而 MySQL 数据 库 都 没有 ; 
MySQL 支持 的 ENUM 和 SET 类 型 ， 这 些 SQL Server 数据 库 不 支持 。 从 某 种 意义 上 说 ， 这 种 差异 是 商 
业 数 据 库 公 司 故意 造成 的 壁垒 ， 是 阻碍 数据 库 市 场 健康 发 展 的 。 


18.4 表 的 导出 和 导入 


MySQL 数据 库 中 的 表 可 以 导出 成 文本 文件 、XML 文件 或 者 HTML 文件 ， 相 应 的 文本 文件 也 可 以 
导入 MySQL 数据 库 中 。 在 数据 库 的 日 常 维护 中 ， 经 常 需要 进行 表 的 导出 和 导入 的 操作 。 本 节 将 介绍 
将 表 内 容 导 出 和 导入 文本 文件 的 方法 。 
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18.4.1 用 SELECT …INTO OUTFILE 导出 文本 文件 


MySQL 中 ,可 以 在 命令 行 窗口 (MySQL Commend Line Client) 中 使 用 SELECT.…INTO OUTFILE 
语句 将 表 的 内 容 导 出 成 一 个 文本 文件 。 其 基本 语法 形式 如 下 。 

SELECT[ 列 名 ] FROM table[WHERE 语句 ] 

INTO OUTFILE ' 目 标 文件 [OPTION]; 

该 语句 分 为 两 个 部 分 。 前 半 部 分 是 一 个 普通 的 SELECT 语句 ， 通 过 这 个 SELECT 语句 来 查询 所 需 
要 的 数据 ; 后 半 部 分 是 导出 数据 的 .其 中 ,目标 文件 "参数 指出 将 查询 的 记录 导出 到 哪个 文件 ，OPTION 
参数 可 以 有 常用 的 几 个 选项 。 介 绍 如 下 。 

(1) FIELDS TERMINATED BY 字符 串 ': 设置 字符 串 为 字段 的 分 隔 符 ， 默 认 值 是 “\t”。 

(2) FIELDS ENCLOSED BY 字符 ': 设置 字符 来 括 上 字段 的 值 。 默 认 情况 下 不 使 用 任何 符号 。 

(3) FIELDS OPTIOINALLY ENCLOSED BY 字符 ': 设置 字符 来 括 上 CHAR、VARCHAR 和 TEXT 
等 字符 型 字段 。 默 认 情况 下 不 使 用 任何 符号 。 

(4) FIELDS ESCAPED BY 字符 ': 设置 转 义 字符 ， 默 认 值 为 “\”。 

(5) LINES STARTING BY 字符 串 ': 设置 每 行 开头 的 字符 ， 默 认 情 况 下 无 任何 字符 。 

(6) LINES TERMINATED BY 字符 串 ': 设置 每 行 的 结束 符 ， 默 认 值 是 “\n”。 

例 18.5 下 面 用 SELECT…INTO OUTFILE 语句 来 导出 test 数据 库 下 order 表 的 记录 。 其 中 , 字段 
之 间 用 “、” 隔 开 ， 字 符 型 数据 用 双 引 号 括 起 来 ， 每 条 记录 以 “>” 开 头 。 命 令 如 下 。〔 实 例 位 置 ， 光 
担 \TMNsM\18\18.5) 

SELECT * FROM test.order INTO OUTFILE 'D:\order.txt 


FIELDS TERMINATED BY \、 ' OPTIONALLY ENCLOSED BY \" 
LINES STARTING BY \>' TERMINATED BY \n'; 


“TERMINATED BY "rn'” 可 以 保证 每 条 记录 占 一 行 。 因 为 Windows 操作 系统 下 “\rn” 才 是 回 

车 换行 。 如 果 不 加 这 个 选项 ， 默 认 情况 只 是 “m”。 用 root 用 户 登 录 到 MySQL 数据 库 中 ， 然 后 执行 上 

述 命令 。 执行 完 后 , 可 以 在 D:\ 下 看 到 一 个 名 为 order.txt 的 文本 文件 。order.txt 中 的 内 容 如 图 18.4 所 示 。 
Toraere 


文件 昌 ”编辑 (E) 格式 (Q) 查看) 帮助 (HH) 
>1、“T 怕 “> 1200、 “1 
> 所、 > 























图 18.4 用 SELECT…INTO OUTFILE 导出 文本 文件 


这 些 记 录 都 是 以 “>” 开 头 ， 每 个 字段 之 间 以 “、” 隔 开 。 而 且 ， 字 符 数据 都 加 上 了 引号 。 
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18.4.2 用 mysqldump 命令 导出 文本 文件 


mysqldump 命令 可 以 备份 数据 库 中 的 数据 。 但 是 ， 备 份 时 是 在 备份 文件 中 保存 了 CREATE 语句 和 
INSERT 语句 。 不 仅 如 此 ，mysqldump 命令 还 可 以 导出 文本 文件 。 其 基本 的 语法 形式 如 下 。 
mysqldump -u root -pPassword -T 目标 目录 dbname table [option]; 


其 中 ，Password 参数 表示 root 用 户 的 密码 ， 密 码 紧 挨 着 -p 选项 ， 目 标 目录 参数 指出 文本 文件 的 
路 径 ，dbname 参数 表示 数据 库 的 名 称 ; table 参数 表示 表 的 名 称 ; option 表示 附件 选项 。 这 些 选项 介 
绍 如 下 。 

(1) --fields-terminated-by= 字 符 串 : 设置 字符 串 为 字段 的 分 隔 符 ， 默 认 值 是 “\t”。 

(2) --fields-enclosed-by= 字 符 : 设置 字符 来 括 上 字段 的 值 。 

(3) --fields-optionally-enclosed-by= 字 符 : 设置 字符 括 上 CHAR、VARCHAR 和 TEXT 等 字符 型 
字段 。 

(4) --fields-escaped-by= 字 符 : 设置 转 义 字符 。 

(5) --lines-terminated-by= 字 符 串 : 设置 每 行 的 结束 符 。 


这 些 选项 必须 用 双 引 号 括 起 来 ， 否 则 ，MySQL 数据 库 系 统 将 不 能 识别 这 几 个 参数 。 

例 18.6 用 mysqldump 语句 来 导出 test 数据 库 下 order 表 的 记录 。 其 中 ， 字 段 之 间 用 “、” 隔 开 ， 
字符 型 数据 用 双 引 号 括 起 来 。 命 令 如 下 。 (实例 位 置 : 光盘 \TMNsN18\18.6) 

mysqldump -u root -p -T D:\ test order "~--lines-terminated-by=\nn" 

"--fields-terminated-by=、" "--fields-optionally-enclosed-by="" 

其 中 ，root 用 户 的 密码 为 111， 密 码 紧 挨 着 -p 选项 。--fields-terminated-by 等 选项 都 用 双 引 号 括 起 
来 。 命 令 执行 完 后 ， 可 以 在 Di\ 下 看 到 一 个 名 为 order.txt 的 文本 文件 和 order.sql 文件 。order.txt 中 的 内 
容 如 图 18.5 所 示 。 














图 18.5 用 MYSQLDUMP 命令 导出 文本 文件 











这 些 记 录 都 是 以 “、” 隔 开 ， 而 且 ， 字 符 数据 都 是 加 上 了 引号 。 其 实 ，mysqldump 命令 也 是 调 
SELECT…INTO OUTFILE 语句 来 导出 文本 文件 的 ; 同时 mysqldump 命令 还 生成 了 student sql 文件 。 这 








个 文件 中 有 表 的 结构 和 表 中 的 记录 。 





导出 数据 时 ， 一 定 要 注意 数据 的 格式 。 通常 每 个 字段 之 间 都 必须 用 分 隔 符 隔 开 ， 可 以 使 用 过 号 
(,) 、 空 格 或 者 制 表 符 (Tab 键 ) 。 每 条 记录 占用 一 行 ， 新 记录 要 从 下 一 行 开始 。 字 符 串 数据 要 使 
用 双 引 号 括 起 来 。 


mysqldump 命令 还 可 以 导出 XML 格式 的 文件 ， 其 基本 语法 如 下 。 

mysqldump-u root -pPassword --xmll|-X dbname table >D:name.xml; 

其 中 ,Password 表示 root 用 户 的 密码 ; 使 用 --xml 或 者 -X 选项 就 可 以 导出 XML 格式 的 文件 ; dbname 
表示 数据 库 的 名 称 ，table 表示 表 的 名 称 ，D:mame.xml 表示 导出 的 XML 文件 的 路 径 。 

例 18.7 使 用 mysqldump 命令 将 数据 表 student 中 的 内 容 导出 到 XML 文件 中 。 效 果 如 图 18.6 所 
示 。《 实 例 位 置 : 光盘 \TMNsI\18\18.7) 





ldump -u root xml test student >D:/student.xml 





图 18.6 在 DOS 命令 窗口 中 的 执行 效果 
生成 的 XML 文件 可 以 在 DD 盘 的 根 目录 下 找到 ， 内 容 如 图 18.7 所 示 。 


tp:/ /www.w3.org/2001/XMLSchema-instance"> 
ablespaces as no INFORMATION_SCHEMA FILES tnble on this server -- 


pdae_time="2012-07- 26 .50r reate_time="2012-07- 
lex jength=-3072" Max data lencth="281474976710655" 
oow_format= Dynamic' Verson="10" Engire="MYISAM Name="student'/> 





<irow> 


图 18.7 生成 的 XML 文件 


18.4.3 用 mysql 命令 导出 文本 文件 











mysql 命令 可 以 用 来 登录 MySQL 服务 器 ， 也 可 以 用 来 还 原 备份 文件 。 同 时 ，mysql 命令 也 可 以 导 
出 文本 文件 。 其 基本 语法 形式 如 下 。 








289 


MySQL 从 入 门 到 精通 


mysql -u root -pPassword -e "SELECT 语句 " dbname >D:/name .txt; 
其 中 ，Password 表示 root 用 户 的 密码 ; 使 用 -e 选项 就 可 以 执行 SQL 语句 : “SELECT 语句 ”用 来 





查询 记录 ; D:/name.txt 表示 导出 文件 的 路 径 。 


例 18.8 下 面 用 mysql 命令 来 导出 test 数据 库 下 student 表 的 记录 ， 命 令 如 下 。 (实例 位 置 : 光盘 


\TMNsI\18\18.8) 


mysql -u root -p111 -e"SELECT * FROM student" test > D:/student2.txt 
在 DOS 命 令 窗 口中 执行 上 述 命 令 ,可 以 将 student 表 中 的 所 用 记录 查询 出 来 ,然后 写 入 到 student2.txt 


文档 中 。student2.txt 被 保存 在 DD 盘 根 目录 下 ， 如 图 18.8 所 示 。 


student2.txt 中 的 内 容 如 图 18.9 所 示 。 


名 称 

访 backup.sql 

转 MingRi logLDF 
男 MingRi3mdf 
团 Studentmdf 张 丽 
司 student.sql 人 


ETE 查看 WO ，，，，，，，，， 
id mame Sex 


可 家 
男 
而 











国 Student logJdf 

[student2b¢t 

国 人 ERp 管 理 系统 rer | 
图 18.8 ”mysql 命令 导出 文本 文件 18.9 文档 内 容 

mysql 命令 还 可 以 导出 XML 文件 和 HIML 文件 。mysql 命令 导出 XML 文件 的 语法 如 下 。 

mysql -u root -pPassword --xmll-X -e "SELECT 语句 " dbname >D:/filename.xml 

其 中 , Password 表示 root 用 户 的 密码 ; 使 用 --xml 或 者 -X 选项 就 可 以 导出 XML 格式 的 文件 ; dbname 











表示 数据 库 的 名 称 ，D:/name.xml 表示 导出 的 XML 文件 的 路 径 。 


中 。 


例如 ， 下 面 的 命令 可 以 将 test 数据 库 中 的 student 表 的 数据 导出 到 名 称 为 student.xml 的 XML 文件 


mysql -u root -p111 --xml -e "SELECT * from student" test >D:/ student.xml 
mysql 命令 导出 HTML 文件 的 语法 如 下 。 
mysql -u root -pPassword --htmll|-H -e "SELECT 语句 " dbname >D:/filename.html 


其 中 ， 使 用 --html 或 者 -也 选项 就 可 以 导出 HIML 格式 的 文件 。 
例如 ， 下 面 的 命令 可 以 将 test 数据 库 中 的 student 表 的 数据 导出 到 名 称 为 student.html 的 HIML 文 





件 中 。 


mysql -u root -p111 --html -e "SELECT * from student" test >D:/student.html 


18.4.4 用 LOAD DATAINFILE 命令 将 文本 文件 导入 到 数据 表 


在 MySQL 中 ,可 以 通过 命令 LOAD DATA INFILE 来 实现 将 指定 格式 的 文本 文件 导入 到 数据 表 中 。 
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LOAD DATA INFILE 命令 的 语法 格式 如 下 。 


LOAD DATA [LOW_PRIORITYICONCURRENT] [LOCAL] INFILE file name INTO TABLE table name 

[OPTION]; 

参数 说 明 如 下 。 

(1) LOW_PRIORITY: 如 果 指 定 LOW_PRIORITY， 则 LOAD DATA 语句 会 被 延迟 ， 直 到 没有 其 
他 的 客户 端正 在 读 取 表 。 

(2) CONCURRENT: 如 果 指 定 CONCURRENT， 则 当 LOAD DATA 正在 执行 时 ， 其 他 线程 可 以 
同时 使 用 该 表 的 数据 。 

(3) LOCAL: 如 果 指 定 了 LOCAL， 则 文件 会 被 客户 主机 上 的 客户 端 读 取 ， 并 被 发 送 到 服务 器 。 
文件 会 被 给 予 一 个 完整 的 路 径 名 称 ， 以 指定 其 确切 的 位 置 。 如 果 给 定 的 是 一 个 相对 路 径 名 称 ， 则 此 名 
称 会 被 理解 为 相对 于 启动 客户 端 时 所 在 的 目录 。 如 果 没 有 指定 LOCAL, 则 文件 必须 位 于 服务 器 主机 上 ， 
并 且 被 服务 器 直接 读 取 。 使 用 LOCAL 的 速度 会 略 慢 些 。 

(4) file name: 用 来 指定 要 导入 的 文本 文件 的 路 径 和 名 称 。 这 个 文件 可 以 手动 创建 ， 也 可 以 使 
其 他 的 程序 创建 。 可 以 使 用 绝对 路 径 〈 如 D:/studentl.txt) ， 也 可 以 不 指定 路 径 ， 直 接 写 上 文件 名 (如 
studentl.txt) ， 这 时 服务 器 将 在 默认 数据 库 目录 中 查找 并 读 取 。 

(5) table name 用 来 指定 需要 导入 数据 的 表 名 ， 该 表 在 数据 库 中 必须 存在 ， 表 结构 必须 与 导入 文 
件 的 数据 一 致 。 

(6) OPTION 用 于 设置 相应 的 选项 ， 其 值 可 以 是 下 面 9 个 值 中 的 任何 一 个 。 

@ FIELDS TERMINATED BY ' 字 符 串 ': 用 于 设置 字段 的 隔 符 为 字符 串 对 象 ， 默 认 值 为 “\t”。 

@ FIELDS ENCLOSED BY ' 字 符 ': 用 于 设置 括 上 字段 值 的 字符 符号 , 默认 情况 下 不 使 用 任何 符号 。 

@ FIELDS OPTIONALLY ENCLOSED BY 字符 : 用 来 设置 括 上 CHAR、VARCHAR 和 TEXT 等 
字段 值 的 字符 符号 ， 默 认 情 况 下 不 使 用 任何 符号 。 

@ FIELDS ESCAPED BY ": 用 于 设置 转 义 字 符 的 字符 符号 ， 默 认 情 况 下 使 用 “\” 字 符 。 

回 LINES STARTING BY ' 字 符 ': 用 来 设置 每 行 开头 的 字符 符号 ， 默 认 情况 下 不 使 用 任何 符号 。 

@ LINES TERMINATED BY ' 字 符 串 ': 用 于 设置 每 行 结 束 的 字符 串 符号 ， 默 认 情况 下 使 用 “\m” 
字符 串 。 

@ IGNORE n LINES: 用 于 指定 忽略 文件 的 前 nm 行 记录 。 

@ (字段 列表 ) : 用 于 实现 根据 字段 列表 中 的 字符 和 顺序 来 加 载 记录 。 

@ SET column=expr: 用 于 设置 列 的 转换 条 件 ， 即 所 指定 的 列 经 过 相应 转换 后 才 会 被 加 载 。 

















在 使 用 该 命令 时 ， 必 须根 据 要 导入 文本 文件 中 ,字段 值 的 分 隔 符 来 指定 使 用 的 分 隔 符 ; 并 且 如 
果 文 本 文件 中 字段 的 顺序 与 数据 表 中 字段 的 顺序 不 完全 一 致 ， 就 需要 使 用 “ (字段 列表 ) ”来 指定 
”加 载 字 段 的 顺序 。 
例 18.9 使 用 LOAD DATA INFILE 命令 , 将 DD 盘 根 目 录 下 的 studentl.txt 文件 中 的 数据 记录 导入 

到 数据 库 test 中 的 studentl 表 中 。 实例 位 置 ， 光盘 \TMsIN18\18.9) 
(1) 准备 一 个 名 称 为 studentl .txt 的 文本 文件 ， 并 放置 在 DD 盘 根 目录 下 ， 内 容 如 图 18.10 所 示 ， 对 
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于 这 个 文件 ， 可 以 使 用 例 18.8 介绍 的 方法 来 进行 导出 。 
(2) 进入 到 MySQL 的 命令 行 窗口 中 ， 输 入 以 下 命令 选择 数据 库 test。 
USE test; 
执行 效果 如 图 18.11 所 示 。 








文件 日“ 所 坦 目 ” 格 式 (DO) 可 看 VJ) 大 二 () 
id name sex zhuanye address 











图 18.10 ”studentl.txt 的 内 容 图 18.11 选择 数据 库 
(3) 在 test 数据 库 中 ， 创 建 一 张 名 称 为 studentl 的 数据 表 ， 表 结构 如 图 18.12 所 示 。 

























Column Name Datatype PK NN UQ BIN UN ZF AL Defout 
id JTJ 加 回 避 
name VARCHAR(20) 避 


sex VARCHAR(4) 避 
zhuanye VARCHAR(20) | 
> address VARCHAR(30) 辣 


] 
] 
] 
age INT(10) 加 回 
| 
] 
] 


图 18.12 数据 表 studentl 的 表 结构 


(4) 执行 LOAD DATA INFILE 命令 ， 将 文本 文件 student1l.txt 中 的 数据 导入 到 表 studentl 中 ， 具 
体 代码 如 下 。 
LOAD DATA INFILE 'D:/student1.txt INTO TABLE student1 
FIELDS TERMINATED BY At 


LINES TERMINATED BY \nn' 
IGNORE 1 LINES; 


执行 效果 如 图 18.13 所 示 ， 数 据 表 studentl 中 的 数据 如 图 18.14 所 示 。 


Inysql> LOAD DATA INFILE ’D: dentl -txt” INIO TABLE student1 
TERMINATED BY 


Warning 





图 18.13 执行 LOAD DATA INFILE 命令 导入 文本 文件 到 数据 表 
(5) 应 用 SELECT 语句 查询 数据 表 studentl 中 的 数据 ， 代 码 如 下 。 
select * from student1; 
执行 结果 如 图 18.14 所 示 。 
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图 18.14 数据 表 studentl 的 数据 


18.4.5 用 mysqlimport 命令 导入 文本 文件 


在 MySQL 中 ， 如 果 只 是 恢复 数据 表 中 的 数据 ， 可 以 在 Windows 的 命令 提示 符 窗口 中 使 用 
mysqllimport 命令 来 实现 。 通 过 mysqlimport 命令 可 以 实现 将 指定 格式 的 文本 文件 导入 到 数据 表 中 。 实 
际 上 , 这 个 命令 提供 了 LOAD DATA INFILE 语句 的 一 个 命令 行 接口 , 它 发 送 一 个 LOAD DATA INFILE 
命令 到 服务 器 来 运行 ， 它 的 大 多 数 选项 直接 对 应 于 LOAD DATA INFILE 命令 。mysqlimport 命令 的 语 
法 格式 如 下 。 


mysqlimport --no-defaults -u root -pPassword -T database file_name [option]; 





人 注意 
这 里 的 -p 与 Password 之 间 是 没有 空格 的 。 


其 中 ，--no-defaults 表示 指定 不 要 从 任何 选项 文件 中 读 取 默 认 选项 ， 这 是 必 选 项 ，Password 参数 表 
示 root 用 户 的 密码 ， 密 码 紧 挨 着 -p 选项 ， 目 标 目录 参数 指出 文本 文件 的 路 径 ，database 参数 表示 要 导 
入 数据 的 数据 库 的 名 称 ; file_name 参数 表示 要 导入 数据 的 文本 文件 名 ; OPTION 用 于 设置 相应 的 选项 ， 
其 值 可 以 是 下 面 几 个 值 中 的 任何 一 个 。 
(1) --fields-terminated-by= 字 符 串 : 用 于 设置 字段 的 分 隔 符 ， 默 认 值 是 “\t”。 
(2) --fields-enclosed-by= 字 符 : 用 于 设置 括 上 字段 值 的 字符 符号 ， 默 认 情 况 下 不 使 用 任何 符号 。 
(3) --fields-optionally-enclosed-by= 字 符 : 用 于 设置 括 上 CHAR、VARCHAR 和 TEXT 等 字符 型 字 
段 值 的 字符 符号 ， 默 认 情况 下 不 使 用 任何 符号 。 
(4) --fields-escaped-by= 字 符 : 用 于 设置 转 义 字符 
(5) --lines-terminated-by= 字 符 串 : 用 于 设置 每 行 的 结束 符 。 
(6) --ignore-lines=n: 用 于 指定 忽略 文件 的 前 mn 行 记 录 。 
例 18.10 使 用 mysqlimport 命令 ,将 DD 盘 根 目录 下 的 studentl.txt 文件 中 的 数据 记录 导入 到 数据 
库 test 中 的 studentl 表 中 。〔 实 例 位 置 : 光盘 \TMNsN\18\18.10) 
(1) 准备 一 个 名 称 为 order.txt 的 文本 文件 ， 并 放置 在 DD 盘 根 目 录 下 ， 内 容 如 图 18.15 所 示 。 
(2) 在 test 数据 库 中 ,创建 一 张 名 称 为 order 的 数据 表 ， 表 结构 如 图 18.16 所 示 。 
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Column Name Datatype PK NN UQ BIN UN ZF AI Default 
于 加 
NULL 
NULL 
NULL 
图 18.15 ”order.txt 的 内 容 图 18.16 数据 表 order 的 表 结 构 


(3) 执行 mysqlimport 命令 ， 将 文本 文件 order.txt 中 的 数据 导入 到 表 order 中 ， 具 体 代 码 如 下 。 





mysqlimport --no-defaults -u root -p111 test D:\order.txt "~lines-terminated-by=\r\in" "--fields-terminated-by=\t" 
"--fields-optionally-enclosed-by=\"" 


执行 效果 如 图 18.17 所 示 。 





图 18.17 执行 mysqlimport 命令 导入 文本 文件 到 数据 表 
(4) 应 用 SELECT 语句 查询 数据 表 order 中 的 数据 ， 代 码 如 下 。 


Use test; 
select * from "order ; 


执行 结果 如 图 18.18 所 示 。 


298 
1988 


12998 
869 
959 





图 18.18 数据 表 order 的 数据 


/ 
说 明 
在 MySQL 中 ， 表 名 和 字符 囊 可 以 用 反 引 号 (也 就 是 数字 1 左 侧 的 那个 键 ) 括 起 来 ， 但 是 这 并 
不 是 必需 的 ， 不 过 ， 如 果 使 用 的 表 名 或 者 字段 是 MySQL 中 的 关键 字 ， 那 么 使 用 反 引 号 将 其 括 起 来 
就 是 必需 的 。 
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18.5 小 结 


本 章 对 备份 数据 库 、 还 原 数据 库 、 数 据 库 迁 移 、 导 出 表 和 导入 表 进 行 了 详细 讲解 ， 备 份 数据 库 和 
还 原 数据 库 是 本 章 的 重点 内 容 。 在 实际 应 用 中 ， 通 常 使 用 mysqldump 命令 备份 数据 库 ， 使 用 mysql 命 
令 还 原 数据 库 。 数 据 库 迁 移 、 导 出 表 和 导入 表 是 本 章 的 难点 。 数 据 迁 移 需要 考虑 数据 库 的 兼容 性 问题 ， 
最 好 是 在 相同 版 本 的 MySQL 数据 库 之 间 迁 移 。 导 出 表 和 导入 表 的 方法 比较 多 ， 希 望 读者 能 够 多 练习 
这 些 方法 的 使 用 。 





18.6 实践 与 练习 


1. 实现 将 数据 表 student 中 的 内 容 导出 到 文本 文件 中 ， 在 生成 文本 文件 时 ， 每 个 字段 之 问 用 逗号 
隔 开 。 每 个 字符 型 的 数据 用 双 引 号 括 起 来 。 而且, 每 条 记录 占 一 行 。 (答案 位 置 : 光盘 \TMNsN18\18.11) 
2. 使 用 mysql 命令 ， 将 表 中 的 记录 导出 到 HTML 文件 中 。 (答案 位 置 ， 光盘 \TMNsN18\18.12) 


和 0。 


MySQL 性 能 优化 


( 僵 ! 视频 讲解 : 10 分 钟 ) 


性 能 优化 是 通过 某 些 有 效 的 方法 提高 MySQL 数据 库 的 性 能 。 性 能 优化 的 目的 
是 为 了 使 MySQL 数据 运行 速度 更 快 、 占 用 的 磁盘 空间 更 小 。 性 能 优化 包括 很 多 方 
面 ， 例 如 优化 查询 速度 、 优 化 更 新 速度 和 优化 MYSQL 服务 器 等 。 本 章 将 介绍 性 能 
优化 的 目的 , 优化 查询 、 优 化 数据 库 结 构 和 优化 多 表 查 询 等 的 方法 ,以 提高 MySQL 
数据 库 的 速度 。 

通过 阅读 本 章 ， 读 者 可 以 : 

了 解 优化 查询 的 方法 

了 解 优化 数据 库 结构 的 方法 
了 解 查询 高 速 缓存 的 方法 
了 解 优 化 多 表 查 询 的 方法 
了 解 优化 表 设 计 的 方法 


各 吾 吾 吾 至 
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19.1 优化 概述 


优化 MySQL 数据库 是 数据 库 管理 员 的 必 备 技能 , 通过 不 同 的 优化 方式 达到 提高 MySQL 数据 库 性 
能 的 目的 。 本 节 将 介绍 优化 的 基本 知识 。 

MySQL 数据 库 的 用 户 和 数据 非常 少 的 时 候 ， 很 难 判断 一 个 MySQL 数据 库 性 能 的 好 坏 。 只 有 当 长 
时 间 运 行 ， 并 且 有 大 量 用 户 进行 频繁 操作 时 ，MySQL 数据 库 的 性 能 才能 体现 出 来 。 例 如 ， 一 个 每 天 有 
几 万 用 户 同时 在 线 的 大 型 网 站 的 数据 库 性 能 的 优 劣 就 很 明显 。 这 么 多 用 户 在 同时 连接 MySQL 数据 库 ， 
并 且 进行 查询 、 插 入 和 更 新 的 操作 ， 如 果 MySQL 数据 库 的 性 能 很 差 ， 很 可 能 无 法 承受 如 此 多 用 户 同 
时 操作 。 试 想 如 果 用 户 查 询 一 条 记录 需要 花费 很 长 时 间 ， 用 户 很 难 会 喜欢 这 个 网 站 。 

因此 ,为 了 提高 MySQL 数据 库 的 性 能 ， 需 要 进行 一 系列 的 优化 措施 。 如 果 MySQL 数据 库 需 要 进 
行 大 量 的 查询 操作 ， 那 么 就 需要 对 查询 语句 进行 优化 。 对 于 耗费 时 间 的 查询 语句 进行 优化 ， 可 以 提高 
整体 的 查询 速度 。 如 果 连 接 MySQL 数据 库 用 户 很 多 ， 那 么 就 需要 对 MySQL 服务 器 进行 优化 ， 否 则 ， 
大 量 的 用 户 同时 连接 MySQL 数据 库 ， 可 能 会 造成 数据 库 系 统 骨 溃 。 

数据 库 管 理 员 可 以 使 用 SHOW STATUS 语句 查询 MySQL 数据 库 的 性 能 。 语 法 形式 如 下 。 

SHOW STATUS LIKE value'; 


其 中 ，value 参数 是 常用 的 几 个 统计 参数 ， 具 体 介 绍 如 下 。 
(1) Connections: 连接 MySQL 服务 器 的 次 数 。 

(2) Uptime: MySQL 服务 器 的 上 线 时 间 。 

(3) Slow_queries: 慢 查 询 的 次 数 。 

(4) Com select: 查询 操作 的 次 数 。 

(5) Com insert: 插入 操作 的 次 数 。 

(6) Com delete: 删除 操作 的 次 数 。 





MySQL 中 存在 查询 InnoDB 类 型 的 表 的 一 些 参数 。 例 如 ，Innodb rows read 参数 表示 SELECT 

语句 查询 的 记录 数 ; Innodb rows inserted 参数 表示 INSERT 语句 插入 的 记录 数 ; Innodb rows_ 

”tpdated 参数 表示 UPDATE 语句 更 新 的 记录 数 ; Innodb rows deleted 参数 表示 DELETE 语句 删除 的 
记录 数 。 





如 果 需 要 查询 MySQL 服务 器 的 连接 次 数 ， 可 以 执行 下 面 的 SHOW STATUS 语句 。 
SHOW STATUS LIKE 'Connections'; 


通过 这 些 参 数 可 以 分 析 MySQL 数据 库 性 能 。 然 后 根据 分 析 结果 ， 进 行 相应 的 性 能 优化 。 


例 19.1 使 用 SHOW STATUS 语句 实现 查看 MySQL 服务 器 的 连接 和 查询 次 数 。 查 看 MySQL 服 
务 器 的 连接 和 查询 次 数 的 语句 执行 效果 如 图 19.1 所 示 。 (实例 位 置 : 光盘 \TMNsI\M9\19.1) 
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nysql> SHOW STATUS LIKE ’Slow_queries’; 


Value } 





图 19.1 查看 MySQL 服务 器 的 连接 和 查询 次 数 
使 用 SHOW STATUS 语句 时 , 可 以 通过 指定 统计 参数 为 Connections、Com select 和 Slow_queries， 
来 实现 显示 MySQL 服务 器 的 连接 数 、 查 询 次 数 和 慢 查 询 次 数 的 功能 ， 关 键 代码 如 下 。 


SHOW STATUS LIKE 'Connections'; 
SHOW STATUS LIKE 'Com_select'; 
SHOW STATUS LIKE 'Slow_queries'; 


19.2 优化 查询 


查询 是 数据 库 最 频繁 的 操作 ， 提 高 查询 速度 可 以 有 效 地 提高 MySQL 数据 库 的 性 能 。 本 节 将 介绍 
优化 查询 的 方法 。 


19.2.1 ”分析 查询 语句 


分 析 查 询 语句 在 前 面 章节 中 都 有 应 用 ， 在 MySQL 中 ， 可 以 使 用 EXPLAIN 语句 和 DESCRIBE 语 

句 来 分 析 查 询 语句 。 
应 用 EXPLAIN 关键 字 分 析 查 询 语句 ， 其 语法 结构 如 下 。 
EXPLAIN ”SELECT 语句 ; 








“SELECT 语句 ”参数 为 一 般 数据 库 查 询 命令 ， 如 SELECT * FROM students。 
例 19.2 ”下面 使 用 EXPLAIN 语 名 分析 一 个 查询 语句 , 其 代码 如 下 。( 实 例 位 置 :光盘 \TMNsIN19\19.2) 
EXPLAIN SELECT * FROM timeinfo; 
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其 运行 结果 如 图 19.2 所 示 。 


1> EXPLAIN SELECT * FROM timeinfo ; 


1 tineinfo 1 ALL 





图 19.2 应 用 EXPLAIN 分 析 查 询 语句 

其 中 各 字段 所 代表 的 意义 如 下 所 示 。 

(1) id 列 : 指出 在 整个 查询 中 SELECT 的 位 置 。 

(2) table 列 : 存放 所 查询 的 表 名 。 

(3) type 列 : 连接 类 型 ， 该 列 中 存储 很 多 值 ， 范 围 从 const 到 ALL。 

(4) possible keys 列 : 指出 为 了 提高 查找 速度 ， 在 MySQL 中 可 以 使 用 的 索引 。 

(5) key 列 : 指出 实际 使 用 的 键 。 

(6) rows 列 : 指出 MySQL 需要 在 相应 表 中 返回 查询 结果 所 检验 的 行 数 ， 为 了 得 到 该 总 行 数 ， 
MySQL 必须 扫描 处 理 整个 查询 ， 再 乘 以 每 个 表 的 行 值 

(7) Extra 列 : 包含 一 些 其 他 信息 ， 设 计 MySQL 如 何 处 理 查询 。 

在 MySQL 中 ， 也 可 以 应 用 DESCRIBE 语句 来 分 析 查 询 语句 。DESCRIBE 语句 的 使 用 方法 与 
EXPLAIN 语法 是 相同 的 ， 这 两 者 的 分 析 结 果 也 大 体 相 同 。 其 中 ，DESCRIBE 的 语法 结构 如 下 。 

DESCRIBE SELECT 语句 ; 





在 命令 提示 符 下 输入 如 下 命 


人 





describe select * from studentinfo; 


其 运行 结果 如 图 19.3 所 示 。 


elect_type ! table ! ! possible_keys ! key ! key_len } x 


一 一 一 + 


SIMPLE 1 timeinfo ! ALL NULL 1 NULL | NULL 





图 19.3 应 用 DESCRIBE 分 析 查 询 语句 


将 图 19.3 与 图 19.2 对 比 ， 读 者 可 以 清楚 地 看 出 ， 其 运行 结果 基本 相同 。 分 析 查 询 也 可 以 应 用 关键 
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字 DESCRIBE。 


/ 
说 明 
DESCRIBE 可 以 缩写 成 DESC. 


19.2.2 ”索引 对 查询 速度 的 影响 


在 查询 过 程 中 使 用 索引 ， 势 必 会 提高 数据 库 查询 效率 ， 应 用 索引 来 查询 数据 库 中 的 内 容 ， 可 以 减 
少 查询 的 记录 数 ， 从 而 达到 查询 优化 的 目的 。 

例 19.3 通过 对 使 用 索引 和 不 使 用 索引 进行 对 比 ， 来 分 析 查 询 的 优化 情况 。〔 实 例 位 置 ， 光 盘 
\TMNsN\M19\19.3) 

首先 ， 分 析 未 使 用 索引 时 的 查询 情况 ， 其 代码 如 下 。 

explain select * from studentinfo where name= 'mrsoft '; 


其 运行 结果 如 图 19.4 所 示 。 





NULL ! NULL 


1 





图 19.4 未 使 用 索引 的 查询 情况 

上 述 结 果 表 明 ， 表 格 字段 rows 下 为 7， 这 意味 着 在 执行 查询 过 程 中 ， 数 据 库存 在 的 7 条 数据 都 被 
查询 了 一 遍 ， 这 样 在 数据 存储 量 小 的 时 候 ， 查 询 不 会 有 太 大 影响 ， 试 想 当 数 据 库 中 存储 庞大 的 数据 资 
料 时 ， 用 户 为 了 搜索 一 条 数据 而 遍历 整个 数据 库 中 的 所 有 记录 ， 将 会 耗费 很 多 时 间 。 现 在 ， 在 name 
字段 上 建立 一 个 名 为 index_name 的 索引 。 创 建 索引 的 代码 如 下 。 

CREATE INDEX index_name ON studentinfo(name); 

上 述 代码 的 作用 是 在 studentinfo 表 的 name 字段 上 添加 索引 。 在 建立 索引 完毕 后 , 然后 再 应 用 关键 
字 EXPLAIN 分 析 执 行情 况 ， 其 代码 如 下 所 示 。 

explain select * from studentinfo where name = 'mrsoft ; 


其 运行 结果 如 图 19.5 所 示 。 
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udentinfo 9 2 Be index_name 


一 一 一 一 一 一 一 一 一 一 一 4 一 一 一 一 一 一 一 一 一 





图 19.5 使 用 索引 后 查询 情况 


从 上 述 结果 可 以 看 出 ， 由 于 创建 的 索引 使 访问 的 行 数 由 7 行 减少 到 1 行 ， 所 以 ， 在 查询 操作 中 ， 
使 用 索引 不 但 会 自动 优化 查询 效率 ， 同 时 也 会 降低 服务 器 的 开销 。 


19.2.3 ”使 用 索引 查询 


在 MySQL 中 ， 索 引 可 以 提高 查询 的 速度 ， 但 并 不 能 充分 发 挥 其 作用 ， 所 以 在 应 用 索引 查询 时 ， 
也 可 以 通过 关键 字 或 其 他 方式 来 对 查询 进行 优化 处 理 。 


1. 应 用 关键 字 LIKE 优化 索引 查询 

例 19.4 下 面 的 示例 应 用 关键 字 LIKE, 并 且 匹 配 字符 串 中 含有 百 分 号 “%” 符 号 , 应 用 EXPLAIN 
语句 执行 如 下 命令 。《〈 实 例 位 置 : 光盘 \ITMNsIN19\19.4) 

EXPLAIN SELECT * FROM studentinfo WHERE name LIKE '%l'; 


其 运行 结果 如 图 19.6 所 示 。 
ysql> explain select #* from studentinfo where name like ’» 


! possible_keys 


SIMPLE tudenti 1 1 NULL ! NULL ! NULL 
g whe 


一 一 一 一 一 一 和 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 也 一 一 一 一 一 一 二 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 | 


row in set (@.22 sec)> 





图 19.6 应 用 关键 字 LIKE 优化 索引 查询 


从 图 19.6 中 可 能 看 2 rows 参数 仍 为 “7” 并 没有 起 到 优化 作用 ， 这 
第 a 分 号 “%” 时 ， 索 引 不 会 被 使 用 ， 如 果 “%” 所 在 匹配 字 入 
则 索引 会 被 正常 使 用 ， 在 命令 提示 竺 中 输入 如 下 命令 。 


是 因为 如 果 匹 配 字符 串 中 ， 
千 串 中 的 位 置 不 是 第 一 位 置 ， 


UH 
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EXPLAIN SELECT * FROM studentinfo WHERE name LIKE 'le%'; 
运行 结果 如 图 19.7 所 示 。 


lect x* from studentinfo where name like ’ 


一 * 一 一 一 一 一 一 一 一 一 | 


1 key_len 


row in set 《8.80 





图 19.7 正常 应 用 索引 的 LIKE 子 句 运行 结果 
2. 查询 语句 中 使 用 多 列 索引 
多 列 索引 在 表 的 多 个 字段 上 创建 一 个 索引 。 只 有 查询 条 件 中 使 用 了 这 些 
引 才 会 被 正常 使 用 。 
应 用 多 列 索 引 在 表 的 多 个 字段 中 创建 一 个 索引 ， 其 命令 如 下 。 
CREATE INDEX index_student_info ON studentinfo(name,sex); 





索 


念 曙 
在 应 用 sex 字段 时 ， 索 引 不 能 被 正常 使 用 。 这 就 意味 着 索引 并 未 在 MySQL 优化 中 起 到 任何 作 
用 , 故 必须 使 用 第 一 字段 name 时 ,索引 才 可 以 被 正常 使 用 ,， 有 兴趣 的 读者 可 以 实际 动手 操作 一 下 。 
这 里 不 再 次 述 .。 


3. 查询 语句 中 使 用 关键 字 OR 

在 MySQL 中， 查询 语句 只 有 包含 关键 字 OR 时 ， 要 求 查询 的 两 个 字段 必须 同 为 索引 ， 如 果 所 搜索 
的 条 件 中 ， 有 一 个 字段 不 为 索引 ， 则 在 查询 中 不 会 应 用 索引 进行 查询 。 其 中 ， 应 用 关键 字 OR 查询 索 
引 的 命令 如 下 。 

SELECT * FROM studentinfo WHERE name='Chris' or sex="M'; 

例 19.5 通过 EXPLAIN 来 分 析 查 询 命令 ， 在 命令 提示 符 中 输入 如 下 代码 。 (实例 位 置 ， 光盘 
VIMNsIN19\19.5) 








EXPLAIN SELECT * FROM studentinfo WHERE name='Chris’ or sex='M'; 
其 运行 结果 如 图 19.8 所 示 。 
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1 index_name-index_student_info ! NULL ! 





图 19.8 ”应 用 关键 字 OR 


从 图 19.8 中 可 以 看 出 ， 由 于 两 个 字段 均 为 索引 ， 故 查询 被 优化 。 如 果 在 子 查询 中 存在 没有 被 设置 
成 索引 的 字段 ， 则 将 该 字段 作为 子 查询 条 件 时 ， 查 询 速度 不 会 被 优化 。 





19.3 ”优化 数据 库 结构 


数据 库 结 构 是 否 合理 ， 需 要 考虑 是 否 存在 见 余 、 对 表 的 查询 和 更 新 的 速度 、 表 中 字段 的 数据 类 型 
否 合理 等 多 方面 的 内 容 。 本 节 将 介绍 优化 数据 库 结 多 的 方法 。 


19.3.1 将 字段 很 多 的 表 分 解 成 多 个 表 


有 些 表 在 设计 en 了 很 多 的 字段 。 这 些 表 中 有 受 的 使 用 频率 很 低 。 当 这 些 表 的 数据 量 很 大 
时 ， 查 询 数据 的 速度 就 会 很 慢 。 本 节 将 介绍 优化 这 种 表 的 方法 。 

对 于 这 种 : he 且 有 些 字段 的 使 用 频率 很 低 的 表 ， 可 以 将 其 分 解 成 多 个 表 。 

例 19.6 F 面 的 学 生 表 中 有 很 多 字段 ， 其 中 在 extra 字 i 学 生 的 备注 信息 。 有 些 备 注 信 息 
的 内 容 特 别 多 , 但 是 , 备注 信息 很 少 使 用 。 这 样 就 可 以 分 解 出 -个 表 , 将 这 个 表 取 名 为 student_extra。 
表 中 存储 两 个 字段 ,分 别 为 id 和 extra。 其 中 ,id 字 段 为 学 生 的 学 学 号 ,extra 字 段 存储 备注 信息 。student_extra 
表 的 结构 如 表 19.9 所 示 。 《实例 位 置 ， 光盘 \TNMNsMN19\19.6) 











图 19.9 将 字段 很 多 的 表 分 解 成 多 个 表 
如 果 需 要 查询 某 个 学 生 的 备注 信息 ， 可 以 用 学 号 〈id) 来 查询 。 如 果 需 要 将 : 


Ee:3 
虫 

六 
下 
浴 
下 
虹 
Wy 
Ey 
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注 信息 同 时 显示 时 ， 可 以 将 student 表 和 student_extra 表 进 行 联 表 查询 ， 查 询 语句 如 下 。 
SELECT * FROM student,student_extra WHERE student.id=student_extra.id; 


通过 这 种 分 解 ， 可 以 提高 student 表 的 查询 效率 。 因 此 ， 遇 到 这 种 字段 很 多 ， 而 且 有 些 字 段 使 用 不 
频繁 的 ， 可 以 通过 这 种 分 解 的 方式 来 优化 数据 库 的 性 能 。 


19.3.2 ”增加 中 间 表 


要 经 常 查询 某 两 个 表 中 的 几 个 字段 。 如 果 经 常 进行 联 表 查 询 ， 会 降低 MySQL 数据 库 的 查 
询 速度 。 对 于 这 种 情况 ， 可 以 建立 中 间 表 来 提高 查询 速度 。 本 节 将 介绍 增加 中 间 表 的 方法 。 

先 分 析 经 常 需要 同时 查询 哪 几 个 表 中 的 哪些 字段 ， 然 后 将 这 些 字 段 建立 一 个 中 间 表 ， 并 将 原来 那 
几 个 表 的 数据 插入 到 中 间 表 中 ， 之 后 就 可 以 使 用 中 间 表 来 进行 查询 和 统计 。 

例 19.7 创建 中 间 表 temp_score 表 保 存 经 常 要 查询 的 学 号 、 姓 名 和 成 绩 等 信息 。 实例 位 置 ， 光 
盘 \TMNsI\M1%\19.7) 

下 面 有 个 学 生 表 student 和 分 数 表 score， 这 两 个 表 的 结构 如 图 19.10 所 示 。 





有 时 需 











o_increment 上 


1 intC18> 


“8.88 sec》 





图 19.10 增加 中 间 表 


实际 中 经 常 要 查 学 生 的 学 号 、 姓 名 和 成 绩 。 根据 这 种 情况 可 以 创建 一 个 temp_score 表 。 temp_score 
表 中 存储 3 个 字段 ， 分 别 是 id、name 和 grade。CREATE 语句 执行 如 下 。 








CREATE TABLE temp_scorelid INT NOT NULL, 
Name VARCHAR(20) NOT NULL, 
grade FLOAT); 


然后 从 student 表 和 score 表 中 将 记录 导入 到 temp_score 表 中 。INSERT 语句 如 下 。 


INSERT INTO temp_score SELECT student.id,student.name,score.grade 
FROM student,score WHERE student.id=score.stu_id; 
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将 这 些 数 据 插入 到 temp_score 表 中 以 后 ， 可 以 直接 从 temp_score 表 中 查询 学 生 的 学 号 、 姓 名 和 成 
绩 。 这 样 就 省 去 了 每 次 查询 时 进行 表 连 接 的 操作 ， 可 以 提高 数据 库 的 查询 速度 。 


19.3.3 ”优化 插入 记录 的 速度 


插入 记录 时 ， 索 引 、 唯 一 性 校 验 都 会 影响 到 插入 记录 的 速度 。 而 且 ， 一 次 插入 多 条 记录 和 多 次 插 
入 记录 所 耗费 的 时 间 是 不 一 样 的。 根据 这 些 情况 ， 分 别 进行 不 同 的 优化 。 本 节 将 介绍 优化 插入 记录 的 
速度 的 方法 。 

1. 禁用 索引 

插入 记录 时 ，MySQL 会 根据 表 的 索引 对 插入 的 记录 进行 排序 。 如 果 插 入 大 量 数据 时 ， 这 些 排序 会 
降低 插入 记录 的 速度 。 为 了 解决 这 种 情况 ， 在 插入 记录 之 前 应 先 禁用 索引 ， 等 到 记录 都 插入 完毕 后 再 
开启 索引 。 禁 用 索引 的 语句 如 下 。 

ALTER TABLE 表 名 DISABLE KEYS; 

重新 开启 索引 的 语句 如 下 。 

ALTER TABLE 表 名 ENABLE KEYS; 

对 于 新 创建 的 表 ， 可 以 先 不 创建 索引 。 等 到 记录 都 导入 以 后 再 创建 索引 。 这 样 可 以 提高 导入 数据 
的 速度 。 

2. 禁用 唯一 性 检查 

插入 数据 时 ，MySQL 会 对 插入 的 记录 进行 校 验 。 这 种 校 验 也 会 降低 插入 记录 的 速度 。 可 以 在 插入 
记录 之 前 禁用 唯一 性 检查 。 等 到 记录 插入 完毕 后 再 开启 。 禁 用 唯一 性 检查 的 语句 如 下 。 

SET UNIQUE_CHECKS=0; 
重新 开启 唯一 性 检查 的 语句 如 下 。 

SET UNIQUE_CHECKS=1; 

3. 优化 INSERT 语句 

插入 多 条 记录 时 ， 可 以 采取 两 种 写 INSERT 语句 的 方式 。 第 一 种 是 一 个 INSERT 语句 插入 多 条 记 
录 。INSERT 语句 的 情形 如 下 。 


INSERT INTO food VALUES 

(NULL, 果 冻 ', 'CC 果冻 厂 ,1.8,'2011, 北京 )， 

(NULL, 咖啡 'CF 咖啡 厂 ,25, '2012', 天津 '), 

(NULL， 奶 糖 , ' 旺 仔 奶 糖 ,15, '2013", "广东 '); 

第 二 种 是 一 个 INSERT 语句 只 插入 一 条 记录 ， 执 行 多 个 INSERT 语句 来 插入 多 条 记录 。INSERT 
语句 的 情形 如 下 。 


INSERT INTO food VALUES(NULL, ,果冻 '，CC 果冻 厂 ',1.8, "2011", "北京 "); 
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INSERT INTO food VALUES(NULL, "咖啡 ', 'CF 咖啡 厂 .25, '2012'，' 天 津 ); 
INSERT INTO food VALUES(NULL, ' 奶 糖 ', ' 旺 仔 奶 糖 ,15,'2013'，' 广 东 "); 
第 一 种 方式 减少 了 与 数据 库 之 间 的 连接 等 操作 ， 其 速度 比 第 二 种 方式 要 快 。 
wa 
-说明 
当 插 入 大 量 数据 时 ,建议 使 用 一 个 INSERT 语句 插入 多 条 记录 的 方式 。 而 且 ， 如果 能 用 LOAD 
DATA INFILE 语句 , 就 尽量 用 LOAD DATA INFILE 语句 。 因 为 LOAD DATA INFILE 语句 导入 数 
据 的 速度 比 INSERT 语句 快 。 


19.3.4 分析 表 、 检 查 表 和 优化 表 


分 析 表 主要 作用 是 分 析 关 键 字 的 分 布 。 检 查 表 主要 作用 是 检查 表 是 否 存在 错误 。 优 化 表 主 要 作用 
是 消除 删除 或 者 更 新 造成 的 空间 浪费 。 本 节 将 介绍 分 析 表 、 检 查 表 和 优化 表 的 方法 。 

1. 分 析 表 

MySQL 中 使 用 ANALYZE TABLE 语句 来 分 析 表 ， 该 语句 的 基本 语法 如 下 。 

ANALYZE TABLE 表 名 仙 表 名 2…]; 

使 用 ANALYZE TABLE 分 析 表 的 过 程 中 , 数据库 系统 会 对 表 加 一 个 只 读 锁 。 在 分 析 期 间 ， 只 能 
读 取 表 中 的 记录 ， 不 能 更 新 和 插入 记录 。ANALYZE TABLE 语句 能 够 分 析 InnoDB 和 MyISAM 类 型 
的 表 。 


例 19.8 下 面 使 用 ANALYZE TABLE 语句 分 析 score 表 , 分 析 结 果 如 图 19.11 所 示 。( 实 例 位 置 : 
光盘 \TMNsI\19\19.8) 





图 19.11 分 析 表 
上 面 结果 显示 了 4 列 信息 ， 详 细 介绍 如 下 。 
(1) Table: 表示 表 的 名 称 。 
(2) Op: 表示 执行 的 操作 。analyze 表示 进行 分 析 操作 ，check 表示 进行 检查 查找 ，optimize 表示 


进行 优化 操作 。 
(3) Msg type: 表示 信息 类 型 ， 其 显示 的 值 通常 是 状态 、 警 告 、 错 误 或 信息 。 
(4) Msg text:: 显示 信息 。 
险 查 表 和 优化 表 之 后 也 会 出 现 这 4 列 信息 。 








306 


第 19 章 “《MySQL 性 能 优化 


2. 检查 表 
MySQL 中 使 用 CHECK TABLE 语句 来 检查 表 .CHECK TABLE 语句 能 够 检查 InnoDB 和 MyISAM 
类 型 的 表 是 否 存在 错误 。 而 且 ， 该 语句 还 可 以 检查 视图 是 否 存 在 错误 。 该 语句 的 基本 语法 如 下 。 
CHECK TABLE 表 名 1[, 表 名 2…][option]; 
其 中 ，option 参数 有 5 个 参数 ， 分 别 是 QUICK、FAST、CHANGED、MEDIUM 和 EXTENDED。 
这 5 个 参数 的 执行 效率 依次 降低 。option 选项 只 对 MyISAM 类 型 的 表 有 效 ， 对 InnoDB 类 型 的 表 无 效 。 
CHECK TABLE 语句 在 执行 过 程 中 也 会 给 表 加 上 只 读 锁 。 
3. 优化 表 
MySQL 中 使 用 OPTIMIZE TABLE 语句 来 优化 表 。 该 语句 对 InnoDB 和 MyISAM 类 型 的 表 都 有 效 。 
但 是 ,OPTILMIZE TABLE 与 拒绝 只 能 优化 表 中 的 VARCHAR、BLOB 或 TEXT 类 型 的 字段 .OPTILMIZE 
TABLE 语句 的 基本 语法 如 下 。 
OPTIMIZE TABLE 表 名 1, 表 名 2-…]; 
通过 OPTIMIZE TABLE 语句 可 以 消除 删除 和 更 新 造成 的 磁盘 碎片 ， 从 而 减少 空间 的 浪费 。 
OPTIMIZE TABLE 语句 在 执行 过 程 中 也 会 给 表 加 上 只 读 锁 。 
A. 
0 涪 明 
如 果 一 个 表 使 用 了 TEXT 或 者 BLOB 这 样 的 数据 类 型 ， 那 么 更 新 、 删 除 等 操作 就 会 造成 磁盘 
空间 的 浪费 ,因为 ,更 新 和 删除 操作 后 ,以 前 分 配 的 磁盘 空间 不 会 自动 收回 .使 用 OPTIMIZE TABLE 
语句 就 可 以 将 这 些 磁盘 碎片 整理 出 来 ， 以 便 以 后 再 利用 。 


19.4 查询 高 速 缓存 


在 MySQL 中 ， 用 户 通 过 SELECT 语句 查询 数据 时 ， 该 操作 将 结果 集 保存 到 一 个 特殊 的 高 级 缓存 
中 ， 从 而 实现 查询 操作 。 首 次 查询 后 ， 当 用 户 再 次 做 相同 查询 操作 时 ，MySQL 即 可 从 高 速 缓存 中 检索 
结果 。 这 样 一 来 ， 既 提高 了 查询 速率 ， 也 起 到 优化 查询 的 作用 。 





19.4.1 检验 高 速 缓存 是 否 开启 


例 19.9 在 MySQL 中 应 用 关键 字 VARIABLES， 以 通配符 形式 查看 服务 器 变量 。 其 代码 如 下 。 
(实例 位 置 : 光盘 \TMNsI\19\19.9) 
SHOW VARIABLES LIKE ' %query_cache %'; 


运行 上 述 代码 ， 其 结果 如 图 19.12 所 示 。 
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row in set 0.85 s 


1> show variab 


query_c 
! query 
! query_ 
! query 
! query ock_invalidate 


rows in set 《0.85 sec》 


图 19.12 检验 高 速 缓 存 是 否 开启 





下 面 对 主要 的 参数 进行 说 明 。 

(1) have_query_cache: 表明 服务 器 在 默认 安装 条 件 下 ， 是 否 已 经 配置 Oe 爱 存 。 

(2) query_cache size: 高 速 缓存 分 配 空间 ， 如 果 该 空间 为 86， 则 证 明 分 配给 高 速 绥 存 空间 的 大 小 
为 86MB 。 如 果 该 值 为 0， 则 表明 查询 高 速 缓存 已 经 关闭 。 

(3) query_cache type: 判断 高 速 缓存 开启 状态 , 其 变量 值 范围 为 0~2。 其 中 当 该 值 为 0 或 OFF 时 ， 
表明 查询 高 速 缓存 已 经 关闭 ， 当 该 值 为 1 或 ON 时 表明 高 速 缓存 已 经 打开 ， 其 值 为 2 或 DEMAND 时 ， 
表明 要 根据 需要 运行 带 有 SQL_CACHE 选项 的 SELECT 语句 ， 提 供 查 询 高 速 缓存 。 








19.4.2 ”使 用 高 速 缓存 


在 MySQL 中 ， 查 询 高 速 缓存 的 具体 语法 结构 如 下 。 

SELECT SQL_CACHE * FROM 表 名 ; 

例 19.10 下面 通过 具体 示例 来 查询 高 速 缓存 运行 中 的 结果 。 在 命令 提示 符 下 输入 以 下 命令 。 ( 实 
例 位 置 ， 光盘 \TMNsN19\19.10) 

SELECT SQL_CACHE * FROM student ; 


其 运行 结果 如 图 19.13 所 示 。 








图 19.13 ”使 用 查询 高 速 缓存 运行 结 
然后 不 使 用 高 速 缓存 查询 该 数据 表 ， 其 结果 如 图 19.14 所 示 。 
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图 19.14 未 使 用 查询 高 速 缓存 运行 结果 


: 行 查询 高 速 缓存 ， 将 会 提高 MySQL 数据 库 的 性 能 。 


如 果 经 党 
/ 
说 明 
一 旦 表 有 变化 , 使 用 这 个 表 的 查询 高 速 缓 存 将 会 失效 ， 且 将 从 高 速 缓存 中 删除 。 这 样 放置 查询 
从 旧 表 中 返回 无 效 数据 。 另 外 ， 不 使 用 高 速 缓存 查找 可 以 应 用 SQL NO_CACHE 关键 字 。 





19.5 优化 多 表 查 询 


在 MySQL 中 ， 用 户 可 以 通过 连接 来 实现 多 表 查 询 ， 在 查询 过 程 中 ， 用 户 将 表 中 的 一 个 或 多 个 共 
同 字段 进行 连接 ， 定 义 查 询 条 件 ， 返 回 统一 的 查询 结果 。 这 通常 用 来 建立 RDBMS 常规 表 之 间 的 关系 。 
在 多 表 查 询 中 ， 可 以 应 用 子 查 询 来 优化 多 表 查 询 ， 即 在 SELECT 语句 中 和 骨 套 其 他 SELECT 语句 。 采 用 
子 查询 优化 多 表 查 询 的 好 处 有 很 多 ， 其 中 ， 可 以 将 分 步 查询 的 结果 整合 成 一 个 查询 ， 这 样 就 不 需要 再 
执行 多 个 单独 查询 ， 从 而 提高 了 多 表 查 询 的 效率 。 

例 19.11 下 面 通过 一 个 实例 来 说 明 如 何 优化 多 表 查 询 ， 首 先 在 命令 提示 符 下 输入 如 下 命令 。( 实 
例 位 置 ， 光 得 \TMNsM\9\19.11) 


select address from student where id=(select id from student_extra wh 
ere name='nihao'); 


其 运行 结果 如 图 19.15 所 示 。 


Inysql> select address 
lextra=’nihao’ >; 


from student where id=(select id from student_extra where 





I row in set (8.83 sec》 


图 19.15 应 用 一 般 SELECT 嵌 套 子 查询 
下 面 应 用 优化 算法 ， 以 便 可 以 优化 查询 速度 。 在 命令 提示 符 下 输入 以 下 


命令 。 
select address from student as stu,student_extra as stu_e where stu.id=stu_e.id and stu_e.extra='nihao" 
以 上 命令 的 作用 是 将 student 和 student_extra 表 分 别 设置 别名 stu、stu_ e， 通 过 两 个 表 的 id 字段 建 
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立 连 接 ， 并 判断 student_extra 表 中 是 否 含有 名 称 为 “nihao” 的 内 容 ， 并 将 地 址 在 屏幕 上 输出 。 该 语句 
已 经 将 算法 进行 优化 ， 以 便 提高 数据 库 的 效率 从 而 实现 查询 优化 的 效果 。 其 运行 结果 如 图 19.16 所 示 。 





图 19.16 应 用 算法 的 优化 查询 


如 果 用 户 希 望 避免 因 出 现 SELECT 赃 套 而 导致 代码 可 读 性 下 降 ， 则 用 户 可 以 通过 服务 器 变量 来 进 
行 优 化 处 理 ， 下 面 应 用 SELECT 嵌 套 方式 来 查询 数据 ， 在 命令 提示 符 中 输入 如 下 命令 。 





select name from student where age> (select avg(age) from student_extra) ; 


其 运行 结果 如 图 19.17 所 示 。 





图 19.17 应 用 SELECT 赚 套 查询 数据 
上 述 合并 两 个 查询 的 速率 将 优 于 子 查询 运行 速率 ， 故 采用 服务 器 变量 也 可 以 优化 查询 。 


19.6 ”优化 表 设计 


在 MySQL 数据 库 中 ， 为 了 优化 查询 ， 使 查询 能 够 更 加 精炼 、 高 效 ， 在 用 户 设 计数 据 表 的 同时 ， 
也 应 该 考虑 一 些 因素 。 

首先 ， 在 设计 数据 表 时 应 优先 考虑 使 用 特定 字段 长 度 ， 后 考虑 使 用 变 长 字段 ， 如 在 用 户 创 建 数据 
表 时 ， 考 虑 创建 某 个 字段 类 型 为 varchar 而 设置 其 字段 长 度 为 255， 但 是 在 实际 应 用 时 ， 该 用 户 所 存储 
的 数据 根本 达 不 到 该 字段 所 设置 的 最 大 长 度 ， 命 令 外 如 设置 用 户 性 别 的 字段 ， 往 往 可 以 用 “M” 表 示 
男性 ，“F” 表 示 女 性 ， 如 果 给 该 字段 设置 长 度 为 varchar(50)， 则 该 字段 占用 了 过 多 列 宽 ， 这 样 不 仅 浪 
费 资源 ， 也 会 降低 数据 表 的 查询 效率 。 适 当 调 整 列 宽 不 仅 可 以 减少 磁盘 空间 ， 同 时 也 可 以 使 数据 在 进 
行 处 理 时 产生 的 IO 过 程 减少 。 将 字段 长 度 设 置 成 其 可 能 应 用 的 最 大 范围 可 以 充分 地 优化 查询 效率 。 

改善 性 能 的 另 一 项 技术 是 使 用 OPTIMIZE TABLE 命令 处 理 用 户 经 常 操作 的 表 .。 频繁 地 操作 数据 库 
中 的 特定 表 会 导致 磁盘 碎片 的 增加 ， 这 样 会 降低 MySQL 的 效率 ， 故 可 以 应 用 该 命令 处 理 经 常 操作 的 
数据 表 ， 以 便于 优化 访问 查询 效率 。 

在 考虑 改善 表 性 能 的 同时 ， 要 检查 用 户 已 经 建立 的 数据 表 ， 划 分 数据 的 优势 在 于 可 以 使 用 户 更 好 












310 


第 19 章 MySQL 性 能 优化 


地 设计 数据 表 ， 但 是 过 多 的 表意 味 着 性 能 降低 ， 故 用 户 应 检查 这 些 表 。 检 查 这 些 表 是 否 有 可 能 整合 为 
一 个 表 中 ， 如 没有 必要 整合 ， 在 查询 过 程 中 用 户 可 以 使 用 连接 ， 如 果 连 接 的 列 采 用 相同 的 数据 类 型 和 
长 度 ， 同 样 可 以 达到 查询 优化 的 作用 。 


~ 
数据 库 表 的 类 型 InnoDB 或 BDB 表 处 理 行 存储 与 MyISAM 或 ISAM 表 的 情况 不 同 . 在 InnoDB 
或 BDB 类 型 表 中 使 用 定 长 列 ， 并 不 能 提高 其 性 能 。 


19.7 小 结 


本 章 对 数据 库 优化 的 含义 和 查看 数据 性 能 参数 的 方法 进行 了 详细 讲解 ， 然 后 介绍 了 优化 查询 的 方 
法 、 优 化 数据 库 结构 的 方法 和 优化 MySQL 服务 器 的 方法 。 优 化 查询 的 方法 和 优化 数据 库 结果 是 本 章 
的 重点 内 容 ， 优 化 查询 部 分 主要 介绍 了 索引 对 查询 速度 的 影响 。 优 化 数据 库 结构 部 分 主要 介绍 了 如 何 
对 表 进 行 优 化 。 本 章 的 难点 是 优化 MySQL 服务 器 ， 因 为 这 部 分 涉及 很 多 MySQL 配置 文件 和 配置 文件 
中 的 参数 。 


19.8 ”实践 与 练习 





1. 实现 在 MySQL 中 使 用 OPTIMIZE TABLE 语句 来 优化 表 。 (答案 位 置 ， 光盘 \TMNsN19\19.12) 
2. 使 用 DESCRIBE 语句 分 析 一 个 查询 语句 。 (答案 位 置 : 光盘 \TMsI\19\19.13) 
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权限 管理 及 安全 控制 


(名 s 视频 讲解 : 10 分 钟 ) 


保护 MySQL 数据 库 的 安全 ， 就 如 同 离开 汽车 时 镇 上 和 车门， 设置 警报 器 。 之 所 
以 这 么 做 ,主要 是 因为 如 果 不 采 取 这 些 基本 但 很 有 效 的 防范 措施 ， 那 么 汽车 或 者 是 
车 中 的 物品 被 盗 的 可 能 性 会 大 大 增加 。 本 章 将 介绍 有 效 保护 MySQL 数据 库 安全 的 
一 些 有 效 措施 。 

通过 阅读 本 章 ， 读 者 可 以 : 

MH ”了解 安全 保护 策略 

MW， 了 解 用 各 种 命令 实现 对 MySQL 数据 库 的 权限 管理 的 方法 

MW 掌握 使 账户 密码 更 安全 的 方法 

站 掌握 状态 文件 和 日 志文 件 
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20.1 安全 保护 策略 概述 


要 确保 MySQL 的 安全 ， 看 看 首先 应 当做 点 儿 什 么 ? 
1. 为 操作 系统 和 所 安装 的 软件 打 补丁 


如 今 打 开 计算 机 的 时 候 ， 都 会 弹出 软件 的 安全 警告 。 虽 然 有 些 时 候 这 些 警 告 会 给 我 们 带 来 一 些 困 
扰 ， 但 是 采取 措施 确保 系统 打上 所 有 的 补丁 是 绝对 有 必要 的 。 利 用 攻击 指令 和 Internet 上 丰富 的 工具 ， 
即使 恶意 用 户 在 攻击 方面 没有 多 少 经 验 ， 也 可 以 毫 无 阻碍 地 攻击 未 打 补 丁 的 服务 器 。 即 使 用 户 在 使 
托管 服务 器 ， 也 不 要 过 分 依赖 服务 提供 商 来 完成 必要 的 升级 ， 相 反 ， 要 坚持 间隔 性 手动 更 新 ， 以 确保 
和 补丁 相关 的 事情 都 被 处 理 妥 当 。 


2. 禁用 所 有 不 使 用 的 系统 服务 


始终 要 注意 在 将 服务 器 放 入 网 络 之 前 ， 已 经 消除 所 有 不 必要 的 潜在 服务 器 攻击 途径 。 这 些 攻 击 往 
往 是 不 安全 的 系统 服务 带 来 的 ， 通 常 运行 在 不 为 系统 管理 员 所 知 的 系统 中 。 简 言 之 ， 如 果 不 打算 使 用 
一 个 服务 ， 就 禁用 该 服务 。 

3. 关闭 端口 

虽然 关闭 未 使 用 的 系统 服务 是 减少 成 功 攻击 可 能 性 的 好 方法 ， 不 过 还 可 以 通过 关闭 未 使 用 的 端口 
来 添加 第 二 层 安 全 。 对 于 专用 的 数据 库 服务 器 ， 可 以 考虑 关闭 除 22 (SSH 协议 专用 ) 、3306 (MySQL 
数据 库 使 用 的 ) 和 一 些 “ 工 具 ” 专 用 的 〈 如 123 (NTP 专用 ) ) 等 端口 号 在 1024 以 下 的 端口 。 简 言 之 ， 
如 果 不 希望 在 指定 端口 有 数据 通信 ， 就 关闭 这 个 端口 。 除 了 在 专用 防火 墙 工具 或 路 由 器 上 做 这 些 调整 
之 外 ， 还 可 以 考虑 利用 操作 系统 的 防火 墙 。 


4. 审计 服务 器 的 用 户 账户 


特别 是 当 已 有 的 服务 器 再 作为 公司 的 数据 库 主机 时 ， 要 确保 禁用 所 有 非特 权 用 户 ， 或 者 最 好 是 全 
部 删除 。 虽 然 MySQL 用 户 和 操作 系统 用 户 完全 无 关 ， 但 他 们 都 要 访问 服务 器 环境 ， 仅 赁 这 一 点 就 可 
能 会 有 意 地 破坏 数据 库 服 务 器 及 其 内 容 。 为 完全 确保 在 审计 中 不 会 有 遗漏 ， 可 以 考虑 重新 格式 化 所 有 
相关 的 驱动 器 ， 并 重新 安装 操作 系统 。 

5. 设置 MySQL 的 root 用 户 密码 

对 所 有 MySQL 用 户 使 用 密码 。 客 户 端 程序 不 需要 知道 运行 它 的 人 员 的 身份 。 对 于 客户 /服务 器 应 
用 程序 ， 用 户 可 以 指定 客户 端 程序 的 用 户 名 。 例 如 ， 如 果 other_user 没有 密码 ， 任 何人 可 以 简单 地 用 
Imysql -u other_ user db _name 冒充 他 人 调用 mysql 程序 进行 连接 。 如 果 所 有 用 户 账户 均 存 在 密码 ， 使 
其 他 用 户 的 账户 进行 连接 将 困难 得 多 。 
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20.2 用户 和 权限 管理 

















MySQL 数据 库 中 的 表 与 其 他 任何 关系 表 没 有 区 别 ， 都 可 以 通过 典型 的 SQL 命令 修改 其 结构 和 数 
据 。 随 着 版 本 3.22.11 的 发 行 ， 可 以 使 用 GRANT 和 REVOKE 命令 。 通 过 这 些 命令 ， 可 以 创建 和 禁用 
用 户 ， 可 以 在 线 授予 和 撤回 用 户 访问 权限 。 由 于 语法 严谨 ， 这 消除 了 由 于 不 好 的 SQL 查询 〈 例 如 ， 忘 
记 在 UPDATE 查询 中 加 入 WHERE 字句 ) 所 带 来 的 潜在 危险 的 错误 。 

在 5.0 版 本 中 ,开发 人 员 向 MySQL 管理 工具 又 增加 了 两 个 新 命令 :CREATE USER 和 DROP USER。 
从 而 能 更 容易 地 增加 新 用 户 、 删 除 和 重 命名 用 户 ， 还 增加 了 第 三 个 命令 RENAME USER 用 于 重 命 名 现 
有 的 用 户 。 















































20.2.1 使 用 CREATE USER 命令 创建 用 户 


CREATE USER 用 于 创建 新 的 MySQL 账户 。 要 使 用 CREATE USER 语句 ， 必 须 拥 有 mysql 数据 
库 的 全 局 CREATE USER 权限 ， 或 拥有 INSERT 权限 。 对 于 每 个 账户 ，CREATE USER 会 在 没有 权限 
的 mysql.user 表 中 创建 一 个 新 记录 。 如 果 账 户 已 经 存在 ， 则 出 现 错误 。 使 用 自选 的 IDENTIFIED BY 
子 句 ， 可 以 为 账户 设置 一 个 密码 。user 值 和 密码 的 设置 方法 和 GRANT 语句 一 样 。 其 命令 的 原型 如 下 
所 示 。 

CREATE USER user [IDENTIFIED BY[PASSWORD 'PASSWORD1] 

[, user [IDENTIFIED BY[PASSWORD 'PASSWORD]… 


例 20.1 应 用 CREATE USER 命令 创建 一 个 新 用 户 ， 用 户 名 为 mrsoft， 密码 为 mr， 其 运行 结果 如 
图 20.1 所 示 。 《实例 位 置 : 光盘 \TMNsI\20\20.1) 





图 20.1 通过 CREATE USER 创建 mrsoft 的 用 户 


20.2.2 使 用 DROP USER 命令 删除 用 户 











如 果 存 在 一 个 或 是 多 个 账户 被 闲置 ， 应 当 考 虑 将 其 删除 ， 确 保 不 会 用 于 可 能 的 违法 的 活动 。 利 
DROP USER 命令 就 能 很 容易 地 做 到 , 它 将 从 权限 表 中 删除 用 户 的 所 有 信息 , 即 来 自 所 有 授权 表 的 账户 
权限 记录 。DROP USER 命令 原型 如 下 所 示 。 

DROP USER user [, user] … 
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SO 四 
DROP USER 不 能 自动 关闭 任何 打开 的 用 户 对 话 。 而 且 ， 如 果 用 户 有 打开 的 对 话 ， 此 时 取消 用 
户 ， 则 命令 不 会 生效 ， 直 到 用 户 对 话 被 关闭 后 才 生 效 。 一 旦 对 话 被 关闭 ， 用 户 也 被 取消 ， 此 用 户 再 
次 试图 登录 时 将 会 失败 。 














例 20.2 应 用 DROP USER 命令 删除 用 户 名 为 mrsoft 的 用 户 ， 其 运行 结果 如 图 20.2 所 示 。 (实例 
位 置 ， 光盘 \TMsI\20\20.2) 







1> DROP USER mrs 
a 


图 20.2 使 用 DROP USER 删除 mrsoft 的 用 户 


20.2.3 使 用 RENAME USER 命令 重 命名 用 户 


RENAME USER 语句 用 于 对 原 有 MySQL 账户 进行 重 命名 。RENAME USER 语句 的 命令 原型 
如 下 。 


RENAME USER old_user TO new_user 
[, old_user TO new_user] … 


$b 注 写 
如 果 旧 账户 不 存在 或 者 新 账户 已 存在 ， 则 会 出 现 错误 。 


例 20.3 应 用 RENAME USER 命令 将 用 户 名 为 mrsoft 的 用 户 重新 命名 为 h, 其 运行 结果 如 图 20.3 
所 示 。 (实例 位 置 ， 光盘 \TMNsIN\20\20.3) 


nysql> RENAME USER mrsoft TO 1h; 


[9 ted 0.08 sec》 





图 20.3 ”使 用 RENAME USER 对 mrsoft 的 用 户 重 命名 
20.2.4 GRANT 和 REVOKE 命令 


GRANT 和 REVOKE 命令 用 来 管理 访问 权限 ， 也 可 以 用 来 创建 和 删除 用 户 ， 但 在 MySQL 5.0.2 中 
可 以 利用 CREATE USER 和 DROP USER 命令 更 容易 地 实现 这 些 任务 。GRANT 和 REVOKE 命令 对 于 
谁 可 以 操作 服务 器 及 其 内 容 的 各 个 方面 提供 了 多 程度 的 控制 ， 从 谁 可 以 关闭 服务 器 ， 到 谁 可 以 修改 特 
定 表 字 段 中 的 信息 都 能 控制 。 表 20.1 中 列 出 了 使 用 这 些 命令 可 以 授予 或 撤回 的 所 有 权限 。 

















315 


MySQL 从 入 门 到 精通 


表 20.1 GRANT 和 REVOKE 管理 权限 
























































权 限 意义 
ALL [PRIVILEGES] 设置 除 GRANT OPTION 之 外 的 所 有 简单 权限 
ALTER 允许 使 用 ALTER TABLE 
ALTER ROUTINE 更 改 或 取消 已 存储 的 子 程序 
CREATE 人 允许 使 用 CREATE TABLE 
CREATE ROUTINE 创建 已 存储 的 子 程序 
CREATE 
TEMPORARY 允许 使 用 CREATE TEMPORARY TABLE 
TABLES 
CREATE USER 人 允许 使 用 CREATE USER、DROP USER、 RENAME USER 和 REVOKE ALL PRIVILEGES 
CREATE VIEW 允许 使 用 CREATE VIEW 
DELETE 允许 使 用 DELETE 
DROP 允许 使 用 DROP TABLE 
EXECUTE 允许 用 户 运 行 已 存储 的 子 程序 
FILE 允许 使 用 SELECT…INTO OUTFILE 和 LOAD DATA INFILE 
INDEX 允许 使 用 CREATE INDEX 和 DROP INDEX 
INSERT 允许 使 用 INSERT 
LOCK TABLES 允许 对 拥有 SELECT 权限 的 表 使 用 LOCK TABLES 
PROCESS 允许 使 用 SHOW FULL PROCESSLIST 
REFERENCES 未 被 实施 
RELOAD 允许 使 用 FLUSH 
REPLICATION : 
EE 允许 用 户 询问 从 属 服务 器 或 主 服务 器 的 地 址 
a 用 于 复制 型 从 属 服务 器 从 主 服务 器 中 读 取 二 进 制 日 志 事件 ) 
SELECT 允许 使 用 SELECT 
SHOW DATABASES | SHOW DATABASES 显示 所 有 数据 库 
SHOW VIEW 允许 使 用 SHOW CREATE VIEW 
SHUTDOWN 允许 使 用 mysqladmin shutdown 
i 允许 使 用 CHANGE MASTER、KILL、PURGE MASTER LOGS 和 SET GLOBAL 语句 ， 
mysqladmin debug 命令 ; 允许 连接 (一 次 ) ， 即 使 已 达到 max_connections 
UPDATE 允许 使 用 UPDATE 
USAGE “无 权限 ”的 同义词 
GRANT OPTION 允许 授予 权限 
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如 果 授 权 表 拥有 含有 mixed-case 数据 库 或 表 名 称 的 权限 记录 , 并 且 lower_case_table_names 系统 变 
量 已 设置 ， 则 不 能 使 用 REVOKE 撤销 权限 ， 必 须 直 接 操纵 授权 表 〈 当 lower_case_table_ names 已 设置 
时 ，GRANT 将 不 会 创建 此 类 记录 ， 但 是 此 类 记录 可 能 已 经 在 设置 变量 之 前 被 创建 了 ) 。 

授予 的 权限 可 以 分 为 多 个 层级 。 

1. 全 局 层级 


全 局 权限 适用 于 一 个 给 定 服务 器 中 的 所 有 数据 库 ， 存 储 在 mysql.user 表 中 。GRANT ALL ON *.* 
和 REVOKE ALL ON *.* 只 授予 和 撤销 全 局 权限 。 
2. 数据 库 层级 
数据 库 权 限 适用 于 一 个 给 定数 据 库 中 的 所 有 目标 ,存储 在 mysql.db 和 mysqlhost 表 中 。GRANT ALL 
ON db_name.* 和 REVOKE ALL ON db_name.* 只 授予 和 撤销 数据 库 权 限 。 
3. 表层 级 
表 权 限 适 用 于 一 个 给 定 表 中 的 所 有 列 ， 存 储 在 mysql.tables priv 表 中 。GRANT ALL ON 
db_name.tbl name 和 REVOKE ALL ON db_name.tbl name 只 授予 和 撤销 表 权 限 。 
4. 列 层级 
列 权 限 适用 于 一 个 给 定 表 中 的 单一 列 , 这 些 权 限 存 储 在 mysql.columns_priv 表 中 。 当 使 用 REVOKE 
时 ， 必 须 指定 与 被 授权 列 相同 的 列 。 
5. 子 程序 层级 
CREATE ROUTINE、ALTER ROUTINE、EXECUTE 和 GRANT 权限 适用 于 已 存储 的 子 程序 。 这 
些 权限 可 以 被 授予 为 全 局 层级 和 数据 库 层级 。 而 且 ， 除 了 CREATE ROUTINE 外 ， 这 些 权限 可 以 被 授 
予 为 子 程序 层级 ， 并 存储 在 mysql.procs_priv 表 中 。 
例 20.4 下面 创建 生理 内 以 此 来 讲解 GRANT 和 REVOKE 命令 的 用 法 。 创 建 一 个 管理 员 ， 
可 以 输入 如 图 20.4 所 示 的 命令 。〔 实 例 位置 : 光盘 \ITMNsI\20\20.4) 
以 上 命令 授予 用 户 名 为 mr、 审查 为 mr 的 用 户 使 用 所 有 数据 库 的 所 有 权限 ， 并 允许 他 向 其 他 人 授 
予 这 些 权 限 。 如 果 不 希望 用 户 在 系统 中 存在 ， 可 以 按 如 图 20.5 所 示 的 方式 撤销 。 




















all privileges,grant 
Ped; 


图 20.4 创建 管理 员 命令 图 20.5 撤销 用 户 命令 
现在 ， 按 如 图 20.6 所 示 的 方式 创建 一 个 没有 任何 权限 的 常规 用 户 。 


Inysql> grant 





t identified by ’nagic123’; 


图 20.6 创建 没有 任何 权限 的 常规 用 户 
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可 以 为 用 户 mrsoft 授予 适当 的 权限 ， 方 式 如 图 20.7 所 示 。 


ct,insert.update.delete,index.alter.create,.drop 





图 20.7 授予 用 户 适当 的 权限 命令 


Af 
说明 
要 完成 对 mrsoft 用 户 授 予 权 限 ， 并 不 需要 指定 mrsoft 的 密码 。 


如 果 认 为 mrsoft 权限 过 高 ， 可 以 按 如 图 20.8 所 示 的 方式 减少 一 些 权 限 。 
当 用 户 mrsoft 不 再 需要 使 用 数据 库 时 ， 可 以 按 如 图 20.9 所 示 的 方式 撤销 所 有 的 权限 。 


> revoke alter.create.drop 
> on hooks.x* 





> from mrsoft; 


图 20.8 减少 权限 的 命令 图 20.9 撤销 用 户 的 所 有 权限 


当 用 户 使 用 GRANT 和 REVOKE 命令 更 改 用 户 权限 后 ， 退 出 MySQL 系统 ， 用 户 使 用 新 账户 
名 登录 MySQL 的 时 候 ， 可 能 会 因为 没有 刷新 用 户 授权 表 而 导致 登录 错误 。 这 是 因为 在 用 户 设置 账 
号 完毕 后 ， 只 有 重新 加 载 授权 表 才 能 使 之 前 设置 的 授权 表 生 效 。 使 用 FLUSH PRIVILEGES 命令 可 
以 重 载 授权 表 。 该 命令 将 在 20.3.1 节 中 讲解 。 

另外 需要 注意 的 是 ， 只 有 如 root 这 样 拥 有 全 部 权限 的 用 户 才 可 以 执行 此 命令 。 当 用 户 重 载 授 
权 表 后 ， 退 出 MySQL 后 ， 使 用 新 创建 的 用 户 名 即 可 正常 登录 MySQL。 


20.3 MySQL 数据 库 安 全 常见 问 


中 
沉 


20.3.1 权限 更 改 何 时 生效 


MySQL 服务 器 启动 的 时 候 以 及 使 用 GRANT 和 REVOKE 语句 的 时 候 ， 服 务 器 会 自动 读 取 grant 
表 。 但 是 ， 既 然 我 们 知道 这 些 权限 保存 在 什么 地 方 以 及 它们 是 如 何 保存 的 ， 就 可 以 手动 修改 它们 。 当 
手动 更 新 它们 的 时 候 ，MySQL 服务 器 将 不 会 注意 到 它们 已 经 被 修改 了 。 

我 们 必须 向 服务 器 指出 已 经 对 权限 进行 了 修改 ， 有 3 种 方法 可 以 实现 这 个 任务 。 可 以 在 MySQL 
命令 提示 符 下 (必须 以 管理 员 的 身份 登录 进入 ) 输入 如 下 命令 。 


flush privileges; 


这 是 更 新 权限 最 常 使 用 的 方法 。 或 者 ， 还 可 以 在 操作 系统 中 运行 : 
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mysqladmin flush-privileges 
或 者 是 
mysqladmin reload 


此 后 ， 当 用 户 下 次 再 连接 的 时 候 ， 系 统 将 检查 全 局 级 别 权限 ; 当下 一 个 命令 被 执行 时 ， 将 检查 数 
据 库 级 别 的 权限 ; 而 表 级 别 和 列 级 别 权限 将 在 用 户 下 次 请 求 的 时 候 被 检查 。 


20.3.2 ”设置 账户 密码 


(1) 可 以 用 mysqladmin 命令 在 DOS 命令 窗口 中 指定 密码 。 
mysqladmin -u user_name -h host_name password "newpwd" 
mysqladmin 命令 重 设 服务 器 为 host name， 且 用 户 名 为 user name 的 用 户 的 密码 ， 新 密码 为 


“newpwd”。 
(2) 通过 set password 命令 设置 用 户 的 密码 。 


set password for 'jeffrey'@'%' = password('biscuit'); 
只 有 以 root 用 户 〈 可 以 更 新 mysql 数据 库 的 用 户 ) 身份 登录 ， 才 可 以 更 改 其 他 用 户 的 密码 。 如 果 
没有 以 匿名 用 户 连接 ， 省 略 for 子 句 便 可 以 更 改 自己 的 密码 。 
set password = password('biscuit'); 
(3) 在 全 局 级 别 下 使 用 GRANT USAGE 语句 (在 *.*) 指定 某 个 账户 的 密码 ， 而 不 影响 账户 当前 
的 权限 。 
GRANT USAGE ON *.* TO 'jeffrey@'%' IDENTIFIED BY "biscuit'; 
(4) 在 创建 新 账户 时 建立 密码 ， 要 为 password 列 提供 一 个 具体 值 。 





mysql -u root mysql 

INSERT INTO user (Host,User,Password) 

-> VALUES(%'jeffrey,PASSWORD(biscuit)): 
mysql> FLUSH PRIVILEGES; 


(5) 更 改 已 有 账户 的 密码 ， 要 应 用 UPDATE 语句 来 设置 password 列 值 。 


mysql -u root mysql 

UPDATE user SET Password = PASSWORD(bagel) 
-> WHERE Host = '%' AND User = francis':; 

FLUSH PRIVILEGES; 
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a 
6 培 明 
(1) 当 使 用 SET PASSWORD 、INSERT 或 者 UPDATE 指定 账户 的 密码 时 ， 必 须 用 
PASSWORDO 函 数 对 它 进 行 加 密 (唯一 的 特例 是 如 果 密 码 为 空 ， 则 不 需要 使 用 PASSWORD0).。 之 
所 以 使 用 PASSWORDO 函 数 是 因为 user 表 以 加 密 方式 保存 密码 ， 而 不 是 明文 。 如 果 采 用 下 面 没有 
进行 加 密 的 方式 设置 密码 ， 代 码 如 下 。 
mysql -u root mysql 
INSERT INTO user (Host,User,Password) 
-> VALUES('%',jeffrey'.'biscuit’): 
mysql> FLUSH PRIVILEGES:; 


结果 是 密码 “biscuit” 保存 到 user 表 后 没有 加 密 。 当 jeffrey 使 用 该 密码 连接 服务 器 时 ， 其 代码 
wp 
mysql -u jeffrey -pbiscuit test 
Access denied 


连接 使 用 的 密码 值 将 被 加 密 ， 并 同 保存 在 user 表 中 的 密码 进行 比较 。 但是， 保存 的 值 为 字符 
囊 'biscuit， 因 此 比较 将 失败 ， 服 务 器 拒绝 连接 。 


(2) 如 果 使 用 GRANT … IDENTIFIED BY 语句 或 mysqladmin password 命令 设置 密码 ， 它 们 
均 会 自动 加 密 密 码 。 在 这 种 情况 下 ， 不 需要 使 用 PASSWORD0) 函 数 对 密码 进行 加 密 。 


20.3.3 ”使 密码 更 安全 


(1) 在 管理 级 别 ， 切 记 不 能 将 mysql.user 表 的 访问 权限 授予 任何 非 管理 账户 。 
(2) 采用 下 面 的 命令 模式 来 连接 服务 器 ， 以 此 来 隐藏 密码 。 命 令 如 下 。 


mysql -u francis -p db_name 
Enter password: ******** 


i 


“*” 字 符 指 示 输 入 密码 的 地 方 ， 输 入 的 密码 是 不 可 见 的 。 因 为 它 对 其 他 用 户 不 可 见 ， 与 在 命令 生 
上 指定 它 相 比 ， 这 样 进入 密码 更 安全 。 

(3) 如 果 想 要 从 非 交互 式 方式 下 运行 一 个 脚本 调用 一 个 客户 端 ， 就 没有 从 终端 输入 密码 的 机 会 。 
其 最 安全 的 方法 是 让 客户 端 程序 提示 输入 密码 或 在 适当 保护 的 选项 文件 中 指定 密码 。 


20.4 状态 文件 和 日 志文 件 


MySQL 数据 目录 里 还 包含 许多 状态 文件 和 日 志文 件 ， 如 表 20.2 所 示 。 这 些 文件 默认 存放 位 置 是 
相应 的 MySQL 服务 器 的 数据 目录 ， 其 默认 文件 名 是 在 服务 器 主机 名 上 增加 一 些 后 组 而 得 到 的 。 
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表 20.2 ”MySQL 的 状态 文件 和 日 志文 件 


























文件 类 型 默 认 名 文件 内 容 
进程 ID 文件 HOSTNAME pid MySQL 服务 器 进程 的 ID 
常规 查询 日 志 HOSTNAME log 连接 / 断 开 连 接 时 间 和 查询 信息 
慢 查 询 日 志 HOSTNAME-slow.log 耗 时 很 长 的 查询 命令 的 文本 
赤 更 日 志 RE 创建 /变更 了 数据 表 的 结构 定义 或 者 修改 了 数据 表 内 
容 的 查询 命令 的 文本 

和 创建 /变更 了 数据 表 的 结构 定义 或 者 修改 了 数据 表 内 
二 进 制 变更 日 志 HOSTNAME-bin nnn 容 的 查询 命令 的 二 进 制 表示 法 
二 进 制 变更 日 志 的 索引 文件 HOSTNAME-binindex 使 用 中 的 “二 进 制 变更 日 志文 件 ” 的 清单 
错误 日 志 HOSTNAME.err “启动 /关机 ”事件 和 异常 情况 


20.4.1 进程 ID 文件 


MySQL 服务 器 会 在 启动 时 把 自己 的 进程 ID 写 入 PID 文件 ， 等 运行 结束 时 又 会 删除 该 文件 。PID 
文件 是 允许 服务 器 本 身 被 其 他 进程 找到 的 工具 。 例 如 ， 如 果 运 行 mysql.server， 在 系统 关闭 时 ， 关 闭 
MySQL 服务 器 的 脚本 检查 PID 文件 以 决定 它 需 要 向 哪个 进程 发 出 一 个 终止 信号 。 





20.4.2 日 志文 件 管理 





默认 情况 下 ， 所 有 日 志 创 建 于 mysqld 数据 目录 中 。 通 过 刷新 日 志 ， 可 以 强制 mysqld 来 关闭 和 重 
新 打开 日 志文 件 〈 或 者 在 某 些 情况 下 切换 到 一 个 新 的 日 志 ) 。 当 执行 一 个 FLUSH LOGS 语句 或 执行 
mysqladmin flush-logs 或 mysqladmin refresh 时 ， 出 现 日 志 刷新 。 如 果 正 使 用 MySQL 复制 功能 ， 从 复制 
服务 器 将 维护 更 多 日 志文 件 ， 被 称 为 接替 日 志 。 日 志文 件 的 类 型 如 表 20.3 所 示 。 


表 20.3 日 志文 件 的 类 型 

















日 志文 件 记 入 文件 中 的 信息 类 型 

错误 日 志 | 。 记录 启动 、 运 行 或 停止 mysqld 时 出 现 的 问题 

常规 查询 日 志 记录 建立 的 客户 端 连接 和 执行 的 语句 

更 新 日 志 记录 更 改 数据 的 语句 。 不 赞成 使 用 该 日 志 

二 进 制 日 志 | 记录 所 有 更 改 数据 的 语句 。 还 用 于 复制 

慢 查 询 日 志 记录 所 有 执行 时 间 超过 long_query_time 秒 的 所 有 查询 或 不 使 用 索引 的 查询 





1. 错误 日 志 
错误 日 志 记载 着 MySQL 数据 库 系统 的 诊断 和 出 错 信息 。 如 果 mysqld 莫名 其 妙 地 “ 死 掉 ” 并 且 








mysqld_ safe 需要 各 





新 启动 它 , mysqld_safe 会 在 错误 日 志 中 写 入 一 条 restarted mysqld 消息 。 如 果 mysqld 
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注意 到 需要 自动 检查 或 者 修复 一 个 表 ， 则 错误 日 志 中 会 写 入 一 条 消息 。 

在 一 些 操作 系统 中 ， 如 果 mysqld“ 死 掉 ”， 错 误 日 志 将 包含 堆栈 跟踪 信息 。 跟 踪 信息 可 以 用 来 确 
定 mysqld“ 死 掉 ” 的 地 方 。 可 以 用 --log-error[=file_ name] 选 项 来 指定 mysqld 保存 错误 日 志文 件 的 位 置 。 
如 果 没 有 指定 file name 值 ，mysqld 使 用 错误 日 志 名 host_name.err， 并 在 数据 目录 中 写 入 日 志文 件 。 如 
果 执 行 FLUSH LOGS, 错误 日 志 用 -old 重新 命名 后 级 ， 并 且 mysqld 创建 一 个 新 的 空 日 志文 件 (如果 未 
给 出 --log-error 选项 ， 则 不 会 重新 命名 ) 。 

如 果 不 指定 --log-error, 或 者 (在 Windows 中 ) 使用--console 选项 , 错误 被 写 入 标准 错误 输出 stderr。 
通常 标准 输出 为 服务 器 的 终端 。 

在 Windows 中 ， 如 果 未 给 出 --console 选项 ， 错 误 输出 总 是 写 入 .err 文件 。 


2. 常规 查询 日 志 


如 果 想 要 知道 mysqld 内 部 发 生 了 什么 ， 应 该 用 --log[=file_name] 或 -1 [file_name] 选 项 启动 它 。 如 果 
没有 指定 fle_name 的 值 ， 默 认 名 是 host_name.log。 所 有 连接 和 语句 被 记录 到 日 志文 件 。 如 果 怀 疑 在 客 
户 端 发 生 了 错误 并 想 确切 地 知道 该 客户 端 发 送 给 mysqld 的 语句 时 ， 该 日 志 可 能 非常 有 用 。 

mysqld 按照 它 接收 的 顺序 记录 语句 到 查询 日 志 ， 这 可 能 与 执行 的 顺序 不 同 。 与 更 新 日 志和 二 进 制 
日 志 不 同 ， 它 们 在 查询 执行 后 ， 任 何 一 个 锁 释 放 前 记录 日 志 (查询 日 志 还 包含 所 有 语句 ， 而 二 进 制 日 
志 不 包含 只 查询 数据 的 语句 ) 。 

服务 器 重新 启动 和 日 志 刷 新 不 会 产生 一 般 的 新 查询 日 志文 件 〈( 尽 管 刷新 关闭 并 重新 打开 一 般 查询 
日 志文 件 ) 。 在 UNIX 操作 系统 中 ， 可 以 通过 下 面 的 命令 重新 命名 文件 并 创建 一 个 新 文件 。 

shell> mv hostname.log hostname-old.log 

shell> mysqladmin flush-logs 


shell> cp hostname-old.log to-backup-directory 
shell> rm hostname-old.log 


在 Windows 中 ， 服 务 器 打开 日 志文 件 期 间 不 能 重新 命名 日 志文 件 。 首 先 ， 必 须 停 止 服务 器 ; 然后 
重新 命名 日 志文 件 ， 最 后 ， 重 启 服务 器 来 创建 新 的 日 志文 件 。 


3. 二 进 制 日 志 


二 进 制 日 志 包含 所 有 更 新 的 数据 或 者 已 经 潜在 更 新 的 数据 〈 例 如 ， 没 有 匹配 任何 行 的 一 个 
DELETE) 的 所 有 语句 。 语 句 以 “事件 ”的 形式 保存 ， 它 描述 数据 更 改 。 

/. 
| 

二 进 制 日 志 已 经 代替 了 老 的 更 新 日 志 ， 更 新 日 志 在 MySQL 5.1 中 不 再 使 用 。 

二 进 制 日 志 还 包含 关于 每 个 更 新 数据 库 的 语句 的 执行 时 间 信 息 ， 但 不 包含 没有 修改 任何 数据 的 语 
句 。 如 果 想 要 记录 所 有 语句 (例如 ， 为 了 识别 有 问题 的 查询 ) ， 应 使 用 一 般 查 询 日 志 。 

二 进 制 日 志 的 主要 目的 是 在 恢复 时 能 够 最 大 可 能 地 更 新 数据 库 ， 因 为 二 进 制 日 志 包含 备份 后 进行 
的 所 有 更 新 。 

二 进 制 日 志 还 用 于 在 主 服务 器 上 记录 所 有 将 发 送 给 从 服务 器 的 语句 。 
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当 用 --log-bin[=file_ name] 选 项 启动 时 ，mysqld 写 入 包含 所 有 更 新 数据 的 SQL 命令 的 日 志文 件 。 如 
果 未 给 出 ie_name 值 ， 默 认 名 为 -bin 后 面 所 跟 的 主机 名 。 如 果 给 出 了 文件 名 ， 但 没有 包含 路 径 ， 则 文 
件 被 写 入 数据 目录 。 如 果 在 日 志 名 中 提供 了 扩展 名 (例如 ，--log-bin=file name.extension) ， 则 扩展 名 
会 被 忽略 。 

mysqld 在 每 个 二 进 制 日 志 名 后 面 添 加 一 个 数字 扩展 名 。 每 次 启动 服务 器 或 刷新 日 志 时 该 数字 则 增 
加 。 如 果 当 前 的 日 志 大 小 达到 max_binlog_size， 还 会 自动 创建 新 的 二 进 制 日 志 。 如 果 正 在 使 用 大 的 事 
务 ， 二 进 制 日 志 超过 max_binlog_size， 事 务 全 写 入 一 个 二 进 制 日 志 中 ， 绝 对 不 要 写 入 不 同 的 二 进 制 日 
志 中 。 为 了 能 够 使 当前 用 户 知道 还 使 用 哪个 不 同 的 二 进 制 日 志文 件 ，mysqld 还 创建 一 个 二 进 制 日 志 索 
引文 件 ， 包 含 所 有 使 用 的 二 进 制 日 志文 件 的 文件 名 。 默 认 情况 下 与 二 进 制 日 志文 件 的 文件 名 相同 ， 扩 
展 名 为 .index。 可 以 用 --log-bin-index[=file_ name] 选 项 更 改 二 进 制 日 志 索 引文 件 的 文件 名 。 当 mysqld 在 
运行 时 ， 不 应 手动 编辑 该 文件 ， 如 果 这 样 做 将 会 使 mysqld 变 得 混乱 。 

可 以 用 RESET MASTER 语句 删除 所 有 二 进 制 日 志文 件 ， 或 用 PURGE MASTER LOGS 只 删除 部 
分 二 进 制 文件 。 

如 果 系 统 正 进行 二 进 制 文件 复制 ， 应 确保 没有 从 服务 器 在 使 用 旧 的 二 进 制 日 志文 件 ， 方 可 删除 它 
们 。 一 种 方法 是 每 天 执行 一 次 mysqladmin flush-logs 并 删除 三 天 前 的 所 有 日 志 。 可 以 手动 删除 , 或 最 好 
使 用 PURGE MASTER LOGS， 该 语句 还 会 安全 地 更 新 二 进 制 日 志 索 引文 件 〈 可 以 采用 日 期 参数 ) 。 

具有 SUPER 权限 的 客户 端 可 以 通过 SET SQL LOG _BIN=0 语句 禁止 将 自己 的 语句 记 入 二 进 制 记 
录 。 可 以 用 mysqlbinlog 实用 工具 检查 二 进 制 日 志文 件 。 

如 果 想 要 重新 处 理 日 志 的 语句 ， 这 很 有 用 。 例 如 ， 可 以 从 二 进 制 日 志 更 新 MySQL 服务 器 ， 方 法 
如 下 。 

shell> mysqlbinlog log-file | mysql -h server_name 


如 果 用 户 正 使 用 事务 ， 必 须 使 用 MySQL 二 进 制 日 志 进 行 备份 ， 而 不 能 使 用 旧 的 更 新 日 志 。 

查询 结束 后 、 锁 定 被 释放 前 或 提交 完成 后 的 事务 ， 则 立即 将 数据 记 入 二 进 制 日 志 ， 这 样 可 以 确保 
按 执行 顺序 记 入 日 志 。 

对 非 事 务 表 的 更 新 执行 完毕 后 立即 保存 到 二 进 制 日 志 中 。 对 于 事务 表 ， 如 BDB 或 InnoDB 表 ， 所 
有 更 改 表 的 更 新 (UPDATE、DELETE 或 INSERT) 被 存 入 缓存 中 ,直到 服务 器 接收 到 COMMIT 语句 。 
在 该 点 ， 当 用 户 执行 完 COMMIT 之 前 ，mysqld 将 整个 事务 写 入 二 进 制 日 志 。 当 处 理事 务 的 线程 启动 
时 ， 它 为 缓冲 查询 分 配 binlog_cache_size 大 小 的 内 存 。 如 果 语 句 大 于 该 值 ， 线 程 则 打开 临时 文件 来 保 
存 事务 。 线 程 结束 后 临时 文件 被 删除 。 

binlog _ cache use 状态 变量 显示 使 用 该 缓冲 区 (也 可 能 是 临时 文件 ) 保存 语句 的 事务 数量 。 
binlog_cache_disk_use 状态 变量 显示 这 些 事务 中 实际 上 有 多 少 必须 使 用 临时 文件 。 这 两 个 变量 可 以 用 于 
将 binlog_cache_size 调节 到 足够 大 的 值 ， 以 避免 使 用 临时 文件 。 

Imax_binlog_ cache size (默认 4GB ) 可 以 用 来 限制 用 来 缓存 多 语句 事务 的 缓冲 区 总 大 小 。 如 果 某 个 
事务 大 于 该 值 ， 将 会 失败 并 执行 回 滚 操作 。 

如 果 正 使 用 更 新 日 志 或 二 进 制 日 志 ， 当 使 用 CREATE … SELECT or INSERT … SELECT 时 ， 并 行 
插入 被 转换 为 普通 插入 。 这 样 通过 在 备份 时 使 用 日 志 可 以 确保 重新 创建 表 的 备份 。 

默认 情况 下 ， 并 不 是 每 次 写 入 时 都 将 二 进 制 日 志 与 硬盘 同步 。 因 此 如 果 操 作 系 统 或 机 器 (不 仅 是 
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MySQL 服务 器 ) 崩溃 ， 有 可 能 二 进 制 日 志 中 最 后 的 语句 丢失 了 。 要 想 防 止 这 种 情况 ， 可 以 使 用 
sync_binlog 全 局 变量 (1 是 最 安全 的 值 ， 但 也 是 最 慢 的 ) ， 使 二 进 制 日 志 在 每 N 次 二 进 制 日 志 写 入 后 
与 硬盘 同步 。 即 使 sync_binlog 设置 为 1， 出 现 崩 溃 时 ， 也 有 可 能 表 内 容 和 二 进 制 日 志 内 容 之 间 存 在 不 
一 致 性 。 例 如 ， 如 果 使 用 InnoDB 表 ，MySQL 服务 器 处 理 COMMIT 语句 ， 它 将 整个 事务 写 入 二 进 制 
日 志 并 将 事务 提交 到 InnoDB 中 。 如 果 在 两 次 操作 之 间 出 现 崩 溃 ， 重 启 时 ， 事 务 被 InnoDB 回 深 , 但 仍 
然 存在 二 进 制 日 志 中 。 可 以 用 --innodb-safe-binlog 选项 解决 该 问题 ， 可 以 增加 InnoDB 表 内 容 和 二 进 制 
日 志 之 间 的 一 致 性 。 


Ne 


在 MySQL 5.1 中 不 需要 --innodb-safe-binlog， 由 于 引入 了 XA 事务 支持 ， 该 选项 作废 了 。 





该 选项 可 以 提供 更 大 程度 的 安全 ， 还 应 对 MySQL 服务 器 进行 配置 ， 使 每 个 事务 的 二 进 制 日 志 
(sync_binlog =1) 和 (默认 情况 为 真 ) InnoDB 日 志 与 硬盘 同步 。 该 选项 的 效果 是 崩溃 后 重启 时 ， 在 滚 
可 事务 后 ,MySQL 服务 器 从 二 进 制 日 志 剪 切 回 滚 的 InnoDB 事 务 。 这 样 可 以 确保 二 进 制 日 志 反馈 InnoDB 
表 的 确切 数据 等 ， 并 使 从 服务 器 与 主 服务 器 保持 同步 〈 不 接收 回 滚 的 语句 ) 。 

注意 ， 即 使 MySQL 服务 器 更 新 其 他 存储 引擎 而 不 是 InnoDB， 也 可 以 使 用 --innodb-safe-binlog。 在 
InnoDB 崩溃 恢复 时 ， 只 从 二 进 制 日 志 中 删除 影响 InnoDB 表 的 语句 /事务 。 如 果 崩 溃 恢 复 时 MySQL 服 
务 器 发 现 二 进 制 日 志 变 短 了 《 即 至 少 缺 少 一 个 成 功 提 交 的 InnoDB 事务 ) ， 如 果 sync_binlog =1 并 且 硬 
盘 / 文 件 系统 的 确 能 根据 需要 进行 同步 《有些 不 需要 ) 则 不 会 发 生 ， 则 输出 错误 消息 ("二 进 制 日 志 < 
名 > 比 期 望 的 要 小 ") 。 在 这 种 情况 下 ， 二 进 制 日 志 不 准确 ， 复 制 应 从 主 服务 器 的 数据 快照 开始 。 


4. 慢 查 询 日 志 


慢 查 询 日 志 记载 着 执行 用 时 较 长 的 查询 命令 ， 这 里 所 说 的 “长 ”是 由 MySQL 服务 器 变量 
long_query time 〈 以 秒 为 单位 ) 定义 的 。 每 出 现 一 个 慢 查询 ，MySQL 服务 器 就 会 给 它 的 slow_queries 
状态 计算 器 加 上 一 个 1。 

用 --log-slow-queries[=file_name] 选 项 启动 时 ,mysqld 写 一 个 包含 所 有 执行 时 间 超 过 long_query_time 
秒 的 SQL 语句 的 日 志文 件 。 

如 果 没 有 给 出 file_name 值 ， 默认 为 主机 名 ， 后缀 为 -slow.log。 如 果 给 出 了 文件 名 , 但 不 是 绝对 路 
径 名 ,文件 则 写 入 数据 目录 。 

语句 执行 完 并 且 所 有 锁 释放 后 记 入 慢 查 询 日 志 。 记 录 顺 序 可 以 与 执行 顺序 不 相同 。 

慢 查 询 日 志 可 以 用 来 找到 执行 时 间 长 的 查询 ， 可 以 用 于 优化 。 但 是 ， 检 查 又 长 又 慢 的 查询 日 志 会 
很 困难 。 要 想 容 易 些 ， 可 以 使 用 mysqldumpslow 命令 获得 日 志 中 显示 的 查询 摘要 来 处 理 慢 查询 日 志 。 

在 MySQL 5.1 的 慢 查询 日 志 中 ， 不 使 用 索引 的 慢 查询 同 使 用 索引 的 查询 一 样 记录 。 要 想 防 止 不 使 
用 索引 的 慢 查询 记 入 慢 查询 日 志 ， 使 用 --log-short-format 选项 。 

在 MySQL 5.1 中 ， 通 过 --log-slow-admin-statements 服务 器 选项 ， 可 以 请 求 将 慢 管理 语句 ， 例 如 
OPTIMIZE TABLE、ANALYZE TABLE 和 ALTER TABLE 写 入 慢 查 询 日 志 。 

用 查询 缓存 处 理 的 查询 不 加 到 慢 查 询 日 志 中 ， 因 为 表 有 和 零 行 或 一 行 而 不 能 从 索引 中 受益 的 查询 也 
不 写 入 慢 查 询 日 志 。 
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5. 日 志文 件 维护 


MySQL 服务 器 可 以 创建 各 种 不 同 的 日 志文 件 ， 从 而 可 以 很 容易 地 看 见 所 进行 的 操作 。 但 是 ， 必 须 
定期 清理 这 些 文件 ， 确 保 日 志 不 会 占用 太 多 的 硬盘 空间 。 

当 启用 日 志 使 用 MySQL 时 ， 可 能 想 要 不 时 地 备份 并 删除 旧 的 日 志文 件 ， 并 告诉 MySQL 开始 记 入 
新 文件 。 在 Linux (Red Hat) 的 安装 上 ， 可 为 此 使 用 mysql-log-rotate 脚本 。 如 果 从 RPM 分 发 安装 
MySQL， 脚 本 应 该 自动 被 安装 了 。 

在 其 他 系统 上 ， 必 须 自己 安装 短 脚 本 ， 可 从 cron 等 入 手 处 理 日 志文 件 。 可 以 通过 mysqladmin 
flush-logs 或 SQL 语句 FLUSH LOGS 来 强制 MySQL 开始 使 用 新 的 日 志文 件 。 

日 志清 空 执行 的 操作 如 下 。 

(1) 如 果 使 用 标准 日 志 〈--log) 或 慢 查 询 日 志 〈--log-slow-queries) ， 关 闭 并 重新 打开 日 志文 件 。 
(默认 为 mysql.log 和 `hostname'`-slow.log) 。 

(2) 如 果 使 用 更 新 日 志 (--log-update) 或 二 进 制 日 志 〈--log-bin) ， 关 闭 日 志 并 且 打 开 有 更 高 序列 
号 的 新 日 志文 件 。 

如 果 只 使 用 更 新 日 志 ， 只 需要 重新 命名 日 志文 件 ， 然 后 在 备份 前 清空 日 志 。 例 如 : 

shell> cd mysql-data-directory 


shell> mv mysql.log mysql.old 
shell> mysqladmin flush-logs 


然后 做 备份 并 删除 mysql.old。 
6. 日 志 失 效 处 理 


激活 日 志 功 能 的 弊病 之 一 是 随 着 日 志 的 增加 而 产生 的 大 量 信息 ， 生 成 的 日 志文 件 有 可 能 会 填 满 整 
个 磁盘 。 如 果 MySQL 服务 器 非常 繁忙 且 需 要 处 理 大 量 的 查询 。 用 户 既 想 保持 有 足够 的 空间 来 记录 
MySQL 服务 器 的 工作 情况 日 志 ， 又 想 防 止 日 志文 件 无 限制 地 增长 ， 就 需要 应 用 一 些 日 志文 件 的 失效 处 
理 技术 。 进 行 日 志 失效 处 理 的 方式 主要 有 以 下 几 种 。 

1) 日 志 轮转 

该 方法 适用 于 常规 查询 日 志和 慢 查 询 日 志 这 些 文件 名 固定 的 日 志文 件 ， 在 日 志 轮 转 时 ， 应 进行 
日 志 刷 新 操作 (mysqladmin flush-logs 命令 或 flush logs 语句 ) ， 以 确保 缓存 在 内 存 中 的 日 志 信息 写 入 
磁盘 。 

日 志 轮 转 的 操作 过 程 是 这 样 的 (假设 日 志文 件 的 名 字 是 log) : 首先 ， 第 一 次 轮转 时 ， 把 log 更 名 
为 log.1， 然 后 服务 器 再 创建 一 个 新 的 log 文件 ; 在 第 二 次 轮转 时 ， 再 把 log.1 更 名 为 log.2， 把 log 更 名 
为 log.1， 然 后 服务 器 再 创建 一 个 新 的 log 文件 ， 如 此 循环 ， 创 建 一 系列 的 日 志文 件 。 当 到 达 日 志 轮转 
失效 位 置 时 ， 下 次 轮转 就 不 再 对 它 进行 更 名 ， 直 接 把 最 后 一 个 日 志文 件 覆盖 掉 。 例 如 ， 如 果 每 天 进行 
一 次 日 志 轮转 并 想 保留 最 后 7 天 的 日 志文 件 ， 就 需要 保留 log.1~log.7 共 7 个 日 志文 件 , 等 下 次 轮转 时 ， 
用 log.6 覆盖 原来 的 log.7 成 新 的 log.7， 原 来 的 log.7 就 自然 失效 。 

日 志 轮 转 的 频率 和 需要 保留 的 老 日 志 时 间 取 决 于 MySQL 服务 器 的 繁忙 程度 〈 服 务 器 越 繁忙 ， 生 
成 的 日 志 信 息 就 越 多 ) 和 用 户 分 配 用 于 存放 老 日 志 的 磁盘 空间 。 

UNIX 系统 允许 对 MySQL 服务 器 已 经 打开 并 正在 使 用 的 当前 日 志文 件 进行 更 名 , 日 志 刷 新 操作 将 
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关闭 当前 日 志文 件 并 打开 一 个 新 日 志文 件 ， 用 原来 的 名 字 创 建 一 个 新 的 日 志文 件 。 文 件 名 固定 不 变 的 
日 志文 件 可 以 用 下 面 这 个 shell 脚本 来 进行 轮转 。 
#1/bin/sh 
# rotate_fixed_logs.sh - rotate MySQL log file that has a fixed name 
# Argument 1:log file name 
if[ $# -ne 11; then 
echo "Usage: $0 logname" 1>&2 
exit 1 
if 
logfile=$1 
myv Slogfile.6 $logfile.7 
mv Slogfile.5 $logfile.6 
mv Slogfile.4 $logfile.5 
mv Slogfile.3 $logfile.4 
mv Slogfile.2 $logfile.3 
myv $logfile.1 $logfile.2 
mv S$logfile $logfile.1 
mysqladmin flush-logs 


这 个 脚本 以 日 志文 件 名 作为 参数 ， 既 可 以 直接 给 出 日 志文 件 的 完整 路 径 名 ， 也 可 以 先进 入 日 志文 
件 所 在 的 目录 再 给 出 日 志文 件 的 文件 名 。 比 如 说 ， 如 果 想 对 /usr/mysqldata 目录 名 为 log 的 日 志 进行 轮 
转 ， 可 以 使 用 下 面 这 条 命令 。 

% rotate_fixed_logs.sh /usr/mysql/data/log 

也 可 以 使 用 下 面 的 命令 。 


% cd/usrmysqldata 
% rotate_fixed_logs.sh log 


为 确保 管理 员 自 己 总 是 存在 权限 对 日 志文 件 进行 更 名 , 最 好 是 在 以 mysqladm 为 登录 名 上 机 时 运行 
这 个 脚本 , 这 里 需要 注意 的 是 , 在 这 个 脚本 里 的 mysqladmin 命令 行 上 没有 给 出 -u 或 -p 之 类 的 连接 选项 
参数 。 

如 果 用 户 已 经 把 执行 mysql 客户 程序 时 要 用 到 的 连接 参数 保存 到 了 mysqladmin 程序 的 my.cnf 选 项 
文件 里 ， 就 不 用 在 这 个 脚本 中 的 mysqladmin 命令 行 上 再 次 给 出 它们 。 

如 果 用 户 没有 使 用 选项 文件 ， 就 必须 使 用 -u 和 -p 选项 告诉 mysqladmin 使 用 哪个 MySQL 账户 (这 
个 MySQL 账户 必须 具备 日 志 刷 新 操作 所 需要 的 权限 ) 去 连接 MySQL 服务 器 。 这 样 ，MySQL 账户 的 
口令 将 会 出 现在 rotate_fixed logs.sh 脚本 的 代码 里 ,所 以 为 了 防止 这 个 脚本 成 为 一 个 安全 漏洞 ， 这 里 建 
议 读者 专门 创建 一 个 除了 能 对 日 志 进 行 刷新 以 外 没有 其 他 任何 权限 的 MySQL 账户 ( 即 一 个 具备 且 仅 
具备 RELOAD 权限 的 MySQL 账户) ， 将 该 账户 的 口令 写 到 脚本 代码 里 ， 最 后 再 将 这 个 脚本 设置 成 只 
允许 mysqladm 用 户 去 编辑 和 使 用 。 下 面 这 条 GRANT 语句 将 以 mrsoft 为 用 户 名 、 以 mrsoftpass 为 口令 
创建 出 一 个 如 上 所 述 的 MySQL 账户 来 。 

GRANT RELOAD ON ** TO lush'@'localhost IDENTIFIED BY 'mrsoftpass'; 


创建 出 这 个 账户 之 后 ， 再 把 rotate_fixed_ logs.sh 脚本 中 的 mysqladmin 命令 行 改写 为 如 下 所 示 的 
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命令 。 
mysqladmin -u mrsoft ~pmrsoftpass mrsoft-logs 


在 Linux 系统 上 的 MySQL 发 行 版 本 中 带 有 一 个 用 来 安装 mysql-log-rotate 日 志 轮转 脚本 的 logrotate 
工具 ， 所 以 不 必 非 得 使 用 rotate_fixed_logs.sh 或 者 自行 编写 其 他 的 类 似 脚本 。 如 用 RPM 安装 ， 则 在 
/usr/share/mysql 目录 ; 如 用 二 进 制 方式 安装 ， 则 在 MySQL 安装 目录 的 support-files 目录 ; 如 用 源码 安 
装 ， 则 在 安装 目录 的 share/mysql 目录 中 。 

Windows 系统 上 的 日 志 轮转 与 UNIX 系统 的 不 太一 样 。 如 果 试图 对 一 个 已 经 被 MySQL 服务 器 打 
开 并 使 用 着 的 日 志文 件 进行 更 名 操作 ， 就 会 发 生 “file in use” (文件 已 被 打开 ) 错误 。 要 在 Windows 
系统 上 对 日 志 进 行 轮转 ， 就 得 先 停止 MySQL 服务 器 ， 然 后 对 文件 进行 更 名 ， 最 后 再 重新 启动 MySQL 
服务 器 ， 在 Windows 系统 上 启动 和 停止 MySQL 服务 器 的 步骤 前 面 已 经 介绍 了 。 下 面 是 一 个 进行 日 志 
更 名 的 批 处 理 文件 。 

@echo off 

REM rotate_fixed_logs.bat - rotate MySQL log file that has a fixed name 

if not "%1" == " goto ROTATE 

@echo Usage: rotate_fixed_logs logname 
goto DONE 

:ROTATE 

set logfile=%1 

erase %logfile%.7 

rename %logfile%.6 %logfile%.7 

rename %logfile%.5 %logfile%.6 

rename %logfile%.4 %logfile%.5 

rename %logfile%.3 %logfile%.4 

rename %logfile%.2 %logfile%.3 

rename %logfile%.1 %logfile%.2 

rename %logfile% %logfile%.1 

:DONE 


这 个 批 处 理 程序 的 用 法 与 rotate_fixed_logs.sh 脚本 差不多 , 它 也 需要 提供 一 个 将 被 轮转 的 日 志文 件 
名 作为 参数 ， 如 下 所 示 。 

C:\>rotate_log c:\mysql\data\log 

或 者 如 下 所 示 。 


ci\>cd\imysql\data 
Ci\> rotate_fixed_logs log 


/ 
NC 培 胡 
在 最 初 几 次 执行 日 志 轮 转 脚本 的 时 候 , 日 志文 件 的 数量 尚未 达到 预 设 的 上 限 值 ， 脚本 会 提示 找 
不 到 某 几 个 文件 ， 这 是 正常 的 。 
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2) 以 时 间 为 依据 对 日 志 进行 失效 处 理 

该 方法 将 定期 删除 超过 指定 时 间 的 日 志文 件 ， 适 用 于 变更 日 志和 二 进 制 日 志 等 文件 名 用 数字 编号 
标识 的 日 志文 件 。 

下 面 是 一 个 用 来 对 以 数字 编号 作为 扩展 名 的 日 志文 件 进行 失效 处 理 的 脚本 。 

#1/usr/bin/perl -w 

# expire_numbered_logs.pl — look through a set of numbered MySQL 


# log files and delete those that are more than a week old. 
# Usage: expire_numbered_logs.pl logfile … 

















Use strict; 
die "Usage: $0 logfile ...\n" if @ARGV == 0; 
my $max_allowed_age = 7; #max allowed age in days 


foreach my $file (@ARGV) #check each argument 
‘ 


unlink ($file) if -e $file && -M Sfile >= $max_allowed_age; 

小 

exit(0); 

以 上 这 个 脚本 是 用 Perl 语言 写 的 。Perl 是 一 种 跨 平 台 的 脚本 语言 ， 用 它 编写 出 来 的 脚本 在 UNIX 
和 Windows 系统 上 皆 可 使 用 。 这 个 脚本 也 需要 提供 一 个 被 轮转 的 日 志文 件 名 作为 参数 , 下面 是 在 UNIX 
系统 上 的 用 法 。 

% expire_numbered_logs.pl /usr/mysql/data/update.[0-9]* 

或 者 是 


% cd/usrmysqldata 
% expire_numbered_logs.pl update.[0-9]* 


如 果 传 递 给 这 个 脚本 的 文件 名 参数 不 正确 ， 就 会 很 危险 。 例如 ， 将“*” 作 为 这 个 脚本 的 文件 名 
参数 ， 即 

% cd/usr/mysql/data 

% expire numbered logs.pl* 


这 样 就 会 把 /usr/mysql/data 目录 里 更 新 时 间 大 于 7 天 的 所 有 文件 (不 仅仅 是 日 志文 件 ) 全 都 删 
除 。 由 于 通过 DatePicker 对 象 获取 到 的 月 份 从 0~ 11， 而 不 是 月 份 中 的 1~ 12， 所 以 需要 将 获取 到 
的 结果 再 加 1， 才 能 代表 真正 的 月 份 。 


3) 镜像 机 制 

将 日 志文 件 镜像 到 所 有 的 从 服务 器 上 ， 就 需要 使 用 镜像 机 制 ， 用 户 必 须知 道 主 服务 器 有 多 少 个 从 
服务 器 ， 哪 些 正 在 运行 ， 并 需 依 次 连接 每 一 个 从 服务 器 ， 同 时 发 出 show slave status 语句 以 确定 它 正 处 
理 主 服务 器 的 哪个 二 进 制 日 志文 件 (语句 输出 列表 的 Master Log File 项 ) ， 只 有 所 有 的 从 服务 器 都 不 
会 用 到 的 日 志文 件 才能 删除 。 例 如 ， 本 地 MySQL 服务 器 是 主 服务 器 ， 它 有 两 个 从 MySQL 服务 器 S1 
和 S2。 在 主 服 务 器 上 有 5 个 二 进 制 日 志文 件 ， 它 们 的 名 字 是 mrlog0.38~mrlog0.42。 

SHOW SLAVE STATUS 语句 在 S1 上 的 执行 结果 如 下 。 
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mysql> SHOW SLAVE STATUS\G 

oar ee rio a 

在 S2 上 的 执行 结果 如 下 。 

mysql> SHOW SLAVE STATUS\G 

Master_Log_File:mrlog.40 

这 样 ， 我 们 就 知道 从 服务 器 仍 在 使 用 的 、 最 低 编号 的 二 进 制 日 志 是 mrlog.40， 而 编号 比 它 更 小 的 
那些 二 进 制 日 志 ， 因 为 不 再 有 从 服务 器 需要 用 到 它们 ， 所 以 已 经 可 以 安全 地 删 掉 。 于 是 ， 连 接 到 主 服 
务 器 并 发 出 下 面 的 语句 : 

mysql> PURGE MASTER LOGS TO 'mrlog.040"; 

在 主 服务 器 上 发 出 的 这 条 命令 将 把 编号 小 于 40 的 二 进 制 日 志文 件 删 除 。 


20.5 小 结 


本 章 对 MySQL 数据 库 的 账户 管理 和 权限 管理 的 内 容 进行 了 详细 讲解 ， 其 中 ， 账 户 管理 和 权限 管 
理 是 本 章 的 重点 内 容 。 这 两 部 分 中 的 密码 管理 、 授 权 和 收回 权限 是 重 中 之 重 , 因为 这 些 内 容 涉 及 MySQL 
数据 库 的 安全 。 和 希望 读者 能 够 认真 学 习 这 部 分 的 内 容 。 


20.6 ”实践 与 练习 


1. 实现 创建 一 个 名 称 为 mr 的 用 户 ， 然 后 再 将 其 删除 。 (答案 位 置 ， 光盘 \TMNsI\20\20.5) 
2. 使 用 set password 命令 将 刚刚 创建 的 mr 用 户 的 密码 设置 为 111。 (答案 位 置 : 光盘 \TMsI20\20.6) 
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(名 s 视频 讲解 : 22 分 钟 ) 


PHP 是 一 种 非常 适合 编写 动态 Web 网 页 的 脚本 语言 ， 用 它 编 写 出 来 的 代码 能 
够 方便 地 嵌入 到 Web 页 面 里 。 当 这 个 Web 页 面 被 访问 时 ， 谈 入 在 其 中 的 PHP 代 
码 就 会 被 执行 并 生成 动态 的 HTML 内 容 ， 而 这 些 内 容 将 作为 Web 页 面 的 一 部 分 被 
送 往 用 户 的 Web 浏览 器 去 显示 。 

通过 阅读 本 章 ， 读 者 可 以 : 

MW 了 解 PHP 操作 MySQL 数据 库 的 步骤 

WI 掌握 通过 PHP 国 数 操作 MySQL 数据 库 的 方法 

MW “掌握 使 用 PHP 操作 MySQL 数据 库 中 的 数据 的 方法 

Wp 掌握 PHP 操作 MySQL 事务 的 方法 

MW 掌握 在 PHP 中 操作 MySQL 的 存储 过 程 的 方法 

MW 了 解 PHP 操作 MySQL 数据 库 出 现 的 常见 问题 及 解决 方法 


第 21 章 PHP 管理 MySQL 数据 库 中 的 数据 


21.1 PHP 语言 概述 


21.1.1 PHP 的 概念 


PHP 是 Hypertext Preprocessor ( 超 文本 预 处 理 器 ) 的 缩写 ， 是 一 种 服务 器 端 、 跨 平台 、HTML 赔 
入 式 的 脚本 语言 。 其 独特 的 语法 混合 了 C 语言 、Java 语言 和 Perl 语言 的 特点 ， 是 一 种 被 广泛 应 用 的 开 
源 式 的 多 用 途 脚 本 语言 ， 尤 其 适合 Web 开发 。 


21.1.2 PHP 的 特点 


PHP 起 源 于 1995 年 ， 由 Rasmus Lerdorf 开发 。 目 前 已 有 超过 2 200 万 个 网 站 、1.5 万 家 公司 、450 
万 程序 开发 人 员 在 使 用 PHP 语言 ， 它 是 目前 动态 网 页 开发 中 使 用 最 为 广泛 的 语言 之 一 。PHP 是 生 于 网 
络 、 用 于 网 络 、 发 展 于 网 络 的 一 门 语言 ， 它 一 诞生 就 被 打上 了 自由 发 展 的 烙印 。 目 前 在 国内 外 有 数 以 
千 计 的 个 人 和 组 织 的 网 站 在 以 各 种 形式 和 各 种 语言 学 习 、 发 展 和 完善 它 ， 并 不 断 地 公布 最 新 的 应 用 和 
研究 成 果 。PHP 能 运行 在 包括 Windows、Linux 等 在 内 的 绝 大 多 数 操作 系统 环境 中 ， 常 与 免费 Web 服 
务 器 软件 Apache 和 免费 数据 库 MySQL 配合 使 用 于 Linux 平台 上 ， 具 有 最 高 的 性 价 比 ， 这 3 种 技术 的 
结合 号 称 “黄金 组 合 ”。 下 面 介绍 PHP 开发 语言 的 特点 。 


1. 速度 快 

PHP 是 一 种 强大 的 CGI 脚本 语言 ， 语 法 混合 了 C、Java、Perl 和 PHP 式 的 新 语法 ， 执 行 网 页 速度 
比 CGI、Perl 和 ASP 更 快 ， 而 且 内 和 骨 Zend 加 速 引擎 ， 性 能 稳定 快速 。 这 是 它 的 第 一 个 突出 的 特点 。 

2. 支持 面向 对 象 

面向 对 象 编程 (OOP) 是 当前 的 软件 开发 趋势 ，PHP 对 OOP 提供 了 良好 的 支持 。 可 以 使 用 OOP 
的 思想 来 进行 PHP 的 高 级 编程 ， 对 于 提高 PHP 编程 能 力 和 规划 好 Web 开发 构架 都 非常 有 意义 。 

3. 实用 性 

由 于 PHP 是 一 种 面向 对 象 的 、 完 全 跨 平 台 的 新 型 Web 开发 语言 , 所 以 无 论 从 开发 者 角度 考虑 还 是 
从 经 济 角 度 考 虑 ， 都 是 非常 实用 的 。PHP 语法 结构 简单 ， 易 于 入 门 ， 很 多 功能 只 需 一 个 函数 就 可 以 实 
现 ， 并 且 很 多 机 构 都 相继 推出 了 用 于 开发 PHP 的 IDE 工具 。 

4. 功能 强大 


PHP 在 Web 项 目 开 发 过 程 中 具有 极其 强大 的 功能 ， 而 且 实现 相对 简单 ， 主 要 表现 在 如 下 几 点 。 

(1) 可 操纵 多 种 主流 与 非 主流 的 数据 库 ， 如 MySQL、Access、SQL Server、Oracle、DB2 等 ， 其 
中 ，PHP 与 MySQL 是 现在 绝 佳 的 组 合 ， 可 以 跨 平台 运行 。 

(2) 可 与 轻 量 级 目录 访问 协议 进行 信息 交换 。 
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(3) 可 与 多 种 协议 进行 通信 ， 包 括 IMAP、POP3、SMTP、SOAP 和 DNS 等 。 
(4) 使 用 基于 POSIX 和 Perl 的 正则 表达 式 库 解析 复杂 字符 串 。 
(5) 可 以 实现 对 XML 文档 进行 有 效 管理 及 创建 和 调用 Web 服务 等 操作 。 


5. 可 选择 性 


PHP 可 以 采用 面向 过 程 和 面向 对 象 两 种 开发 模式 ， 并 向 下 兼容 ， 开 发 人 员 可 以 从 所 开发 网 站 的 规 
模 和 日 后 维护 等 多 角度 考虑 ， 以 选择 所 开发 网 站 应 采取 的 模式 。 

PHP 进行 Web 开发 过 程 中 使 用 最 多 的 是 MySQL 数据 库 。PHP 5.0 以 上 版 本 中 不 仅 提供 了 早期 
MySQL 数据 库 操纵 函数 ， 而 且 提 供 了 MySQLi 扩展 技术 对 MySQL 数据 库 的 操纵 ， 这 样 开发 人 员 可 以 
从 稳定 性 和 执行 效率 等 方面 考虑 操纵 MySQL 数据 库 的 方式 。 


6. 成 本 低 


PHP 具有 很 好 的 开放 性 和 可 扩展 性 ， 属 于 自由 软件 ， 其 源 代码 完全 公开 ， 任 何 程序 员 为 PHP 扩展 
附加 功能 非常 容易 。 在 很 多 网 站 上 都 可 以 下 载 到 最 新 版 本 的 PHP。 目 前 ，PHP 主要 是 基于 Web 服务 器 
运行 的 , 支持 PHP 脚本 运行 的 服务 器 有 多 种 , 其 中 最 有 代表 性 的 为 Apache 和 IIS。PHP 不 受 平 台 束缚 ， 
可 以 在 UNIX、Linux 等 众多 版 本 的 操作 系统 中 架设 基于 PHP 的 Web 服务 器 。 采 用 
Linux+Apache+PHP+MySQL 这 种 开源 免费 的 框架 结构 可 以 为 网 站 经 营 者 节省 很 大 一 笔 开 支 。 


7. 版 本 更 新 速度 快 

与 数 年 才 更 新 一 次 的 ASP 相 比 ，PHP 的 更 新 速度 要 快 得 多 ， 因 为 PHP 几乎 每 年 更 新 一 次 。 
8. 模板 化 

实现 程序 逻辑 与 用 户 界面 分 离 。 

9. 应 用 范围 广 


目前 在 互联 网 有 很 多 网 站 的 开发 都 是 通过 PHP 语言 来 完成 的 ， 如 搜狐 、 网 易 和 百度 等 ， 在 这 些 知 
名 网 站 的 创作 开发 中 都 应 用 到 了 PHP 语言 。 











21.1.3 ”PHP 的 工作 原理 


PHP 是 基于 服务 器 端 运行 的 脚本 程序 语言 ， 实 现 数据 库 和 网 页 之 问 的 数据 交互 。 

一 个 完整 的 PHP 系统 由 以 下 几 个 部 分 构成 。 

(1) 操作 系统 : 网 站 运行 服务 器 所 使 用 的 操作 系统 。PHP 不 要 求 操作 系统 的 特定 性 ， 其 跨 平 台 的 
特性 允许 PHP 运行 在 任何 操作 系统 上 ， 如 Windows 和 Linux 等 。 

(2) 服务 器 : 搭建 PHP 运行 环境 时 所 选择 的 服务 器 。PHP 支持 多 种 服务 器 软件 ， 包 括 Apache、 
Is 等 。 

(3) PHP 包 : 实现 对 PHP 文件 的 解析 和 编译 。 

(4) 数据 库 系 统 : 实现 系统 中 数据 的 存储 。 PHP 支持 多 种 数据 库 系统 ,包括 MySQL、SQL Server、 
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Oracle 及 DB2 等 。 

(5) 浏览 器 : 浏览 网 页 。 由 于 PHP 在 发 送 到 浏览 器 的 时 候 已 经 被 解析 器 编译 成 其 他 的 代码 ， 所 以 
PHP 对 浏览 器 没有 任何 限制 。 

在 图 21.1 中 , 完整 地 展示 了 用 户 通过 浏览 器 访问 PHP 网 站 系统 的 全 过 程 ， 可 以 更 加 清晰 地 理 清 它 
们 之 间 的 关系 。 





图 21.1 PHP 的 工作 原理 


MT 

PHP 工作 原理 如 下 。 

(1) PHP 的 代码 传递 给 PHP 包 ， 请 求 PHP 包 进 行 解析 并 编译 。 

(2) 服务 器 根据 PHP 代码 的 请 求 读 取 数 据 库 。 

(3) 服务 器 与 PHP 包 共 同根 据 数 据 库 中 的 数据 或 其 他 运行 变量 ， 将 PHP 代码 解析 成 普通 的 
HIML 代码 。 

(4) 解析 后 的 代码 发 送 给 浏览 器 ， 浏 览 器 对 代码 进行 分 析 获 取 可 视 化 内 容 。 

(5) 用 户 通过 访问 浏览 器 浏览 网 站 内 容 。 在 Android 4.0 中 ， 采 用 默认 的 主题 (Theme.Holo ) 
时 ，android:prompt 属性 看 不 到 具体 的 效果 ， 但 是 采用 Theme.Black 时 ， 就 可 以 看 到 在 弹出 的 下 拉 
框 上 将 显示 该 标题 。 


21.1.4 ”PHP 结合 数据 库 应 用 的 优势 


在 实际 应 用 中 ，PHP 的 一 个 最 常见 的 应 用 就 是 与 数据 库 结合 。 无 论 是 建设 网 站 还 是 设计 信息 系 
统 ， 都 少不了 数据 库 的 参与 。 广 义 的 数据 库 可 以 理解 成 关系 型 数据 库 管理 系统 、XML 文件 ， 甚 至 文 
本 文件 等 。 

PHP 支持 多 种 数据 库 , 而 且 提供 了 与 诸多 数据 库 连 接 的 相关 函数 或 类 库 。 一 般 来 说 ,PHP 与 MySQL 
是 比较 流行 的 一 个 组 合 。 该 组 合 的 流行 不 仅 是 因为 它们 都 可 以 免费 获取 ， 更 多 的 是 因为 PHP 内 部 对 
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MySQL 数据 库 的 完美 支持 。 
当然 ， 除 了 使 用 PHP 内 置 的 连接 函数 以 外 ， 还 可 以 自行 编写 函数 来 间接 存 取 数 据 库 。 这 种 机 制 给 
程序 员 带 来 了 很 大 的 灵活 性 。 


21.2 PHP 操作 MySQL 数据 库 的 基本 步骤 


与 其 他 语言 类 似 , PHP 操作 MySQL 数据 库 的 过 程 一 般 分 为 5 步 ， 分 别 为 连接 MySQL 数据 库 服 务 
器 、 选 择 数 据 库 、 执 行 SQL 语句 、 关 闭 结果 集 以 及 断 开 与 MySQL 服务 器 的 连接 ， 如 图 21.2 所 示 。 


连 摘 MySQL 服务 器 | 
| 
Ea 选择 VySQ1 妆 所 库 。 忆 疙 | 
请 求 | mg 
一 一 ee [> 
+ WE 人 请求 


时 
关闭 结果 集 


断 开 与 ysQL 服 务 器 连接 


图 21.2 PHP 操作 MySQL 数据 库 的 步 又 


下 面 将 对 图 21.2 中 的 5 个 步 又 进行 具体 介绍 。 

(1) 连接 MySQL 服务 器 。 

应 用 mysql_connect0 函 数 建 立 与 MySQL 服务 器 的 连接 ， 并 返回 一 个 连接 标识 ， 在 以 后 对 MySQL 
服务 器 进行 操作 时 ， 可 以 根据 这 个 连接 标识 定位 不 同 的 连接 。 

(2) 选择 数据 库 。 

应 用 mysql_select_ db0 函 数 选择 MySQL 数据 库 服 务 器 上 的 数据 库 ， 并 与 该 数据 库 建 立 连 接 。 

(3) 执行 SQL 语句 。 

在 选择 的 数据 库 中 应 用 mysql_query0 函 数 执行 SQL 语句 。 对 数据 的 操作 主要 包括 以 下 5 种 方式 。 

@ 查询 数据 : 应 用 SELECT 语句 实现 数据 的 查询 功能 。 

@ 显示 数据 : 应 用 SELECT 语句 显示 数据 的 查询 结果 。 

图 插入 数据 : 应 用 INSERT 语句 向 数据 库 中 插入 数据 。 

@ 更 新 数据 : 应 用 UPDATE 语句 修改 数据 库 中 的 记录 。 

@@ 删除 数据 : 应 用 DELETE 语句 删除 数据 库 中 的 记录 。 

(4) 关闭 结果 集 。 

数据 库 操作 完成 后 ， 需 要 关闭 结果 集 ， 以 释放 系统 资源 。 

mysql_free_result($result); 

















NSSQL 数据 库 服务 器 
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GE 

如 果 在 多 个 网 页 中 都 要 频繁 进行 数据 库 访问 , 则 可 以 建立 与 数据 库 服 务 器 的 持续 连接 来 提高 
效率 。 因 为 每 次 与 数据 库 服 务 器 的 连接 需要 较 长 的 时 间 和 较 大 的 资源 开销 ,持续 的 连接 相对 来 说 
会 更 有 效 。 建 立 持续 连接 的 方法 就 是 在 数据 库 连 接 时 ， 调 用 函数 mysql pconnect0 〇 代替 
Imysql connectO 函 数 。 建 立 的 持续 连接 在 本 程序 结束 时 ， 不 需要 调用 mysql close0) 来 关闭 。 下 次 
程序 在 此 执行 mysql pconnectO 函 数 时 ， 系 统 自 动 直接 返回 已 经 建立 的 持续 连接 的 ID 号 ， 而 不 再 
去 真 的 连接 数据 库 。 


(5) 断 开 与 MySQL 服务 器 的 连接 。 

每 使 用 一 次 mysql_connect0 或 mysql_query0 函 数 ， 都 会 消耗 系统 资源 。 这 在 少量 用 户 访问 Web 网 
站 时 影响 不 明显 ， 但 如 果 用 户 连接 超过 一 定数 量 ， 就 会 造成 系统 性 能 的 下 降 ， 甚 至 死机 。 为 了 避免 这 
种 现象 的 发 生 ， 在 完成 数据 库 的 操作 后 ， 可 以 应 用 mysql_close0 函 数 关闭 与 MySQL 服务 器 的 连接 ， 以 
节省 系统 资源 。 


21.3 使 用 PHP 操作 MySQL 数据 库 


根据 21.2 节 中 介绍 的 PHP 操作 MySQL 数据 库 的 步骤 ， 下 面 详细 讲解 每 个 步骤 是 如 何 实现 的 ， 都 
应 用 了 哪些 函数 和 方法 。 


21.3.1 应 用 mysql_connect() 函 数 连接 MySQL 服务 器 


PHP 操作 MySQL 数据 库 , 首先 要 建立 与 MySQL 数据 库 的 连接 , PHP 实现 与 数据 库 连 接 相 对 简便 ， 
只 需 使 用 mysql_connect0 函 数 即 可 ， 函 数 语 法 如 下 。 


resource mysql_connect ( [string server [, string username [, string password [, bool new_link [, int 
client_flags]]]]] ) 


mysql_connect0 函 数 用 于 打开 一 个 到 MySQL 服务 器 的 连接 ， 如 果 成 功 则 返回 一 个 MySQL 连接 标 
识 ， 失 败 则 返回 false。 该 函数 的 参数 如 表 21.1 所 示 。 


表 21.1 mysql_connect() 函 数 的 参数 说 明 
参数 说 明 








MySQL 服务 器 。 可 以 包括 端口 号 ， 如 “hostname:port”; 或 者 到 本 地 套 接 字 的 路 径 ， 如 对 于 localhost 
SeIVeT 的 “:/path/to/socket”。 如 果 PHP 指令 mysqLdefault host 未 定义 〈 默 认 情 况 ) ， 则 默认 值 是 
“localhost:3306” 





Usemame ”| 用 户 名 。 默 认 值 是 服务 器 进程 所 有 者 的 用 户 名 





password ”| 密码 。 默 认 值 是 空 密码 
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续 表 








如 果 用 同样 的 参数 再 次 调用 mysql_connectO 函 数 ， 将 不 会 建立 新 连接 ， 而 将 返回 已 经 打开 的 连接 标 
识 。 参 数 new_link 改变 此 行为 并 使 mysql_ connectO 函 数 总 是 打开 新 的 连接 ， 即 使 mysql_connectO 函 
数 曾 在 前 面 被 用 同样 的 参数 调用 过 

client flags 参数 可 以 是 以 下 常量 的 组 合 : MYSQL_ CLIENT SSL，MYSQL_CLIENT_ COMPRESS， 
MYSQL CLIENT IGNORE SPACE 或 MYSQL CLIENT INTERACTIVE 


new_link 





client flags 


例如 ， 使 用 mysql_connect0 函 数 连接 本 地 MySQL 服务 器 ， 代 码 如 下 。 


<?php 
$conn = mysql_connect("localhost", "root", "root") or die(" 连 接 数 据 库 服 务 器 失败 ! ".mysql_error()); 
pe 


例 21.1 应 用 mysql_connect0 函 数 创 建 与 MySQL 服务 器 的 连接 : MySQL 数据 库 服务 器 地 址 为 
127.0.0.1， 用 户 名 为 root， 密 码 为 root， 代 码 如 下 。《〔 实 例 位 置 ， 光盘 \TMNsIN21\21.1) 


<?php 

$host = "127.0.0.1"; /MySQL 服务 器 地 址 
$userName = "root"; // 用 户 名 

$password = "root"; /密码 


if ($connIlD = mysql_connect($host, $userName, $password)){ 
/建立 与 MySQL 数据 库 的 连接 ， 并 弹出 提示 对 话 框 
echo "<script language='javascript'>alert(' 数 据 库 连接 成 功 ! ');</script>"; 
}else{ 
echo "<script language='javascript'>alert(' 数 据 库 连接 失败 ! '");</script>"; 
上 
?> 


运行 上 述 代 码 ， 如 果 在 本 地 计算 机 中 安装 了 MySQL 数据 库 ， 并 且 root 让 
用 户 名 为 root， 密 码 为 root， 则 会 弹出 如 图 21.3 所 示 的 对 话 框 。 
为 了 方便 查询 因为 连接 问题 而 出 现 的 错误 , 采用 die0 函 数 生成 错误 处 
理 机 制 ， 使 用 mysql_error0 函 数 提取 MySQL 函数 的 错误 文本 。 如 果 没 有 rn 
出 错 ， 则 返回 空 字符 串 ; 如 果 浏 览 器 显示 “Waming: mysql_connectO…?” 
的 字样 时 ,说明 是 数据 库 连接 的 错误 ， 这 样 就 能 迅速 地 发 现 错误 位 置 ， 及 图 21.3 数据库 连接 成 功 
时 改正 。 
SC 
在 mysql_connectO 函 数 前 面 添加 符号 “@”,， 用 于 限制 这 个 命令 的 出 错 信息 的 显示 。 如 果 函 数 
调用 出 错 ， 将 执行 or 后 面 的 语句 。die0 辑 数 表 示 向 用 户 输出 引号 中 的 内 容 后 ， 程 序 终 止 执行 。 这 
样 是 为 了 防止 数据 库 连 接 出 错时 ， 用 户 看 到 一 堆 英名 其 妙 的 专业 名 词 ， 而 是 提示 定制 的 出 错 信 息 。 
但 在 调试 时 不 要 屏蔽 出 错 信 息 ， 避 免 出 错 后 难以 找到 问题 。 


外 sagmielaszbl 
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21.3.2 ”应 用 mysql_select_db() 函 数 选择 MySQL 数据 库 


成 功 与 MySQL 数据 库 建立 连接 后 ， 需 要 选择 MySQL 数据 库 服务 器 中 指定 的 数据 库 。PHP 中 使 用 
mysql_select db0 函 数 实现 数据 库 的 选择 功能 ， 该 函数 的 语法 格式 如 下 。 

bool mysql_select db ( string database_name [, resource link_identifier] ) 

mysql_select_db0 函 数 用 于 设 定 与 指定 的 连接 标识 符 所 关联 的 服务 器 上 的 当前 激活 数据 库 。 如 果 没 
有 指定 连接 标识 符 ， 则 使 用 上 一 个 打开 的 连接 。 如 果 没 有 打开 的 连接 ， 本 函数 将 无 参数 调用 
mysql_connect0 函 数 来 尝试 打开 一 个 使 用 。 其 后 的 每 个 mysql_query0 函 数 调用 都 会 作用 于 当前 激活 数 
据 库 。 该 函数 的 参数 说 明 如 表 21.2 所 示 。 


表 21.2 mysql_select_db() 函 数 的 参数 说 明 

















database name 必要 参数 ， 用 户 指定 要 选择 的 数据 库 名称 
link identifier 可 选 参数 ， 数 据 库 连接 ID， 如果 省 略 该 参数 ， 则 默认 为 最 近 一 次 与 数据 库 建立 的 连接 





例如 ， 与 本 地 MySQL 服务 器 中 的 db_database21 数据 库 建立 连接 ， 代 码 如 下 。 


<?php 

$conn=mysql_connect("localhost","root","root"); /| 连接 mysql 数据 库 服务 器 
$select=mysql_select_db("db_database21",$conn); /连接 服务 器 中 的 db_database21 
if($selectX{ // 判 断 是 否 连 接 成 功 


echo "数据 库 连接 成 功 !"; 
例 21.2 首先 使 用 mysql_connect0 函 数 建 立 与 MySQL 数据 库 的 连接 并 返回 数据 库 连接 ID， 然 后 
使 用 mysql_select_db0 函 数 选择 MySQL 数据 库 服 务 器 中 名 为 db_database21 的 数据 库 ， 实 现代 码 如 下 。 
(实例 位 置 ， 光盘 \TMNs121\21.2) 








<?php 

$host = "127.0.0.1"; /MySQL 服务 器 地 址 

$userName = "root"; /用户 名 

S$password = "root"; /密码 

S$dbName = "db_database21"; /数据 库 

$connlD = mysql_connect($host, $userName, $password);  // 建 立 与 MySQL 数据 库 服务 器 的 连接 
if(mysql_select_db($dbName, $connID)}{ /选择 数据 库 


echo "数据 库 选 择 成 功 ! 
Jelse{ 
echo "数据 库 选择 失败 !"; 
» 


Ye 
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运行 上 述 代 码 ， 如 果 本 地 MySQL 数据 库 服 务 器 中 存在 名 为 db_database21 的 数据 库 ， 将 在 页 面 中 
显示 如 图 21.4 所 示 的 提示 信息 。 











种 httpy/ 5 
GOWNe) rp/nocahosts2ae 262indephp -1 [x)|P BRR P| 
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次 | 昌 |- « 愿 htpy- x 从 ~- 国 - 忆 出 mp 
数据 库 选择 成 功 ! 加 
完成 达 本 地 Intranet | 保护 模式 : 禁用 各 ” 成 100% ~ | 

一 = 














图 21.4 数据 库 选择 成 功 
21.3.3 ”应 用 mysql_query() 函 数 执行 SQL 语句 


成 功 选择 MySQL 数据 库 服 务 器 中 的 数据 库 后 ， 即 可 对 所 选 数 据 库 中 的 数据 表 进 行 查询 、 更 改 以 
及 删除 等 操作 ，PHP 使 用 mysql_query0 函 数 就 可 以 实现 上 述 所 有 操作 ， 操 作 极其 简便 ,说 明 在 PHP 底 
层 进行 了 复杂 的 封装 ， 而 提供 给 上 层 开发 人 员 一 种 简便 的 编程 模式 ， 这 也 是 PHP 操作 简便 的 体现 和 应 
用 广泛 的 原因 。mysql_query0 函 数 的 语法 格式 如 下 。 


resource mysql_query ( string query [, resource link_identifier] ) 
mysql_query0 函 数 用 于 执行 一 条 查询 语句 ， 该 函数 的 参数 说 明 如 表 21.3 所 示 。 
表 21.3 mysql_query() 函 数 的 参数 说 明 


参数 说 明 

字符 串 类 型 ， 传 入 的 是 SQL 的 指令 ， 包 括 插入 数据 〈insert) 、 修 改 记 录 (update) 、 删 除 
a 记录 (delete》、 查 询 记录 (select) 
on 资源 类 型 ， 传 入 的 是 由 mysql_connect0 函 数 或 mysql_pconnect0 函 数 返回 的 连接 号 。 如 果 省 
link identfier 





略 该 参数 ， 则 会 使 用 最 后 一 个 打开 的 MySQL 数据 库 连 接 

例如 ， 向 会 员 信息 表 tb_user 中 插入 一 条 会 员 记 录 ，SQL 语句 的 代码 如 下 。 
Sresult=mysql_query("insert into tb_user values(mr,root)",$conn); 

例如 ， 修 改 会 员 信息 tb_user 表 中 的 会 员 记 录 ，SQL 语句 的 代码 如 下 。 
Sresult=mysql_query("update tb_user set name='1x' where id='01",$conn); 

例如 ， 删 除 会 员 信息 tb_user 表 中 的 一 条 会 员 记 录 ，SQL 语句 的 代码 如 下 。 
$result=mysql_query("delete from tb_user where name='mr",$conn); 

例如 ， 查 询 会 员 信息 tb_user 表 中 name 字段 值 为 mr 的 记录 ，SQL 语句 的 代码 如 下 。 
Sresult=mysql_query("select * from tb_user where name='mr", $conn); 

上 面 的 SQL 语句 代码 都 是 将 结果 赋 给 变量 $result。 
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例 21.3 查询 学 生 信息 表 中 学 生 的 成 绩 信息 ， 代 码 如 下 。 


<?php 

$host = "127.0.0.1"; 

$userName = "root"; 

$password = "root"; 

$dbName = "db_database21"; 

$connlD = mysql_connect($host, $userName, $password); 
mysql_select_db($dbName, $connID); 

mysql_query("set names utf8"); 

echo "<table border=\"1px\" align=\"center\"> 


中 的 数据 


《实例 位 置 ， 光盘 \TMNsI21\21.3) 


/MySQL 数据 库 服务 器 
/用 户 名 

/密码 

/数据 库 名 

/连接 MySQL 数据 库 
/| 选择 MySQL 数据 库 
/设置 字符 集 


/| 执行 查询 
// 获 取 结 果 集 并 输出 查询 结果 


<tr> 
<td> 学 号 </td> 
<td> 姓 名 </td> 
<td> 班 级 </td> 
<td> 语 文 </td> 
<td> 数 学 </td> 
<td> 英 语 </td> 
</tr>"; 
$query = mysql_query("select sno, sname, class, chinese, math ,english from tb_student", $connID); 
while($result = mysql_fetch_array($query)) 
' 
echo "<tr> 
<td>".$result["sno"]."</td> 
<td>".$result["sname"]."</td> 
<td>".$result["class"]."</td> 
<td>".$result["chinese"]."</td> 
<td>".$result["math"]."</td> 
<td>".$result["english"]."</td> 
<htr>", 
内 
echo "</table>"; 
?> 


运行 上 述 实例 ， 结 果 如 图 21.5 所 示 。 
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高 Wi 增强 | -| < 医 httpy.. x 偷 ~ 园 ~ 响 -~ FE ” 
学 号 ”| 姓名 | 班级 ”| 语文 数学 英语 
0312315 | 刘 ** 03 级 3 班 p85 l89 56 引 
0312316 | 黄 ** 03 织 6 班 835 96 195 
间 全 本 地 Intranet | 保护 楼 式 :禁用 种 ~ 以 100% ~ 











图 21.5 查询 学 生成 绩 
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21.3.4 ”应 用 mysql_fetch_array() 函 数 将 结果 集 返 回 到 数组 中 











使 用 mysql_query0 函 数 执行 SELECT 语句 时 ， 成 功 将 返回 查询 结果 集 ， 返 回 结果 集 后 ， 使 
mysql_fetch_array0 函 数 可 以 获取 查询 结果 集 信息 ， 并 放 入 到 一 个 数组 中 ， 函 数 语法 如 下 。 
array mysql_fetch_array ( resource result [, int result_type] ) 


其 中 ， 参 数 result 是 资源 类 型 的 参数 ， 要 传 入 的 是 由 mysql_ query0 函 数 返 回 的 数据 指针 ， 参 数 
result_type 是 可 选项 ， 设 置 结 果 集 数组 的 表述 方式 ， 默 认 值 是 MYSQL_ BOTH。 其 可 选 值 如 下 。 

(1) MYSQL ASSOC: 表示 数组 采用 关联 索引 。 

(2) MYSQL NUM: 表示 数组 采用 数字 索引 。 

(3) MYSQL BOTH: 同时 包含 关联 和 数字 索引 的 数组 。 

例 21.4 按 员工 编号 以 模糊 查询 的 方式 查询 员工 信息 ， 并 显示 全 部 查询 结果 。 (实例 位 置 ， 光 盘 
\TMNslN21\21.4) 

具体 实现 步骤 如 下 。 

(1) 建立 与 MySQL 数据 库 的 连接 ， 并 返回 数据 库 连 接 ID， 代 码 如 下 。 


<?php 























$connID=mysql_connect("localhost","root","root"); /建立 与 数据 库 的 连接 
mysql_select_db("db_database21", $connlD); /选择 数据 库 
mysql_query("set names utf8"); /设置 字符 集 

?> 


(2) 建立 查询 信息 录入 表单 ， 表 单 及 表单 元 素 如 表 21.4 所 示 。 
表 21.4 ”员工 信息 录入 表单 






name="formml" method="post" action="<?php echo $_ 
SERVER[PHP SELF]?>" 

name= number type="text" id="number" 
type="hidden" name="flag" value="1" 
name="submit" type="submit" value=" 提 交 " 












录入 员工 编号 
判断 表单 是 否 提交 
“提交 ”按钮 


(3) 使 用 $_POST 全 局 数组 接收 表单 提交 的 flag 元 素 的 值 ， 并 使 用 issetO 函 数 判断 是 否 已 经 设置 
了 该 元 素 的 值 ， 如 果 已 设置 则 说 明 已 经 提交 了 表单 ， 然 后 采用 模糊 查询 的 方式 查询 所 有 与 查询 关键 字 
相 匹 配 的 员工 信息 ， 并 使 用 while 循环 将 查询 结果 显示 出 来 ， 代 码 如 下 。 
<?php 
if(isset($_POST["flag"])X{ 
$query=mysql_query("select * from tb_employee where number like '%".$_POST['number"]."%"); 


if($queryX{ 
while($myrow=mysql_fetch_array($query)){ 
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<tr> 
<td align="center"” bgcolor="#FFFFFF" class="STYLE4"><span class="STYLE2"><?php echo 
$myrow[numberl;? > </span></td> 
<td align="center" bgcolor="#FFFFFF" class="STYLE4"><span class="STYLE2"><?php echo 
$myrow[name];?> </span></td> 
<td height="23" align="center" bgcolor="#FFFFFF" class="STYLE4"><span class="STYLE2"><?php echo 
$myrow [tel];?></span></td> 
<td height="23" align="center" bgcolor="#FFFFFF" class="STYLE4"><span class="STYLE2"><?php echo 
$myrow [address];?></span></td> 
</tr> 
<?php 


} 
} 


> 
运行 该 实例 ， 在 员工 查询 信息 录入 表单 中 输入 员工 编号 ， 然 后 单 击 “ 提 交 ” 按 钮 ， 即 可 以 模糊 查 
询 的 方式 查询 出 所 有 与 查询 关键 字 相 匹配 的 员工 信息 ， 如 图 21.6 所 示 。 
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图 21.6 查询 员工 信息 


21.3.5 “应 用 mysql_fetch_object() 函 数 从 结果 集中 获取 一 行 作为 对 象 

















21.3.4 节 中 讲解 了 应 用 mysql_fetch_array0 函 数 来 获取 结果 集中 的 数据 。 除 了 这 个 方法 以 外 ， 应 
mysql_fetch_object0 函 数 也 可 以 轻松 实现 这 一 功能 , 下 面 通过 同一 个 实例 的 不 同方 法 来 体验 一 下 这 两 个 
函数 在 使 用 上 的 区 别 。 首 先 介 绍 mysql_fetch_object0 函 数 。 语 法 如 下 。 























object mysql_fetch_object ( resource result ) 


mysql_fetch_object0 函 数 和 mysql_fetch_array0 函 数 类 似 ， 只 有 一 点 区 别 : 即 前 者 返回 一 个 对 象 而 
不 是 数组 ， 即 该 函数 只 能 通过 字段 名 来 访问 数组 。 访 问 结果 集中 行 的 元 素 的 语法 结构 如 下 。 
$row->col_name /icol_name 为 列 名 ，$row 代表 结果 集 














例如 ， 如 果 从 某 数据 表 中 检索 id 和 name 值 ， 可 以 用 $row->id 和 $row-> name 访问 行 中 的 元 素 值 。 
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本 函数 返回 的 字段 名 是 区 分 大 小 写 的 ， 这 是 初学 者 学 习 时 最 容易 忽视 的 问题 。 


例 21.5 使 用 mysql fetch_objectO 函 数 获取 查询 到 图 书 的 信息 。《〈 实 例 位置 ; 光盘 \TMNsI121\21.5) 
具体 开发 步骤 如 下 。 
(1) 建立 图 书 查询 表单 ， 表 单 及 表单 元 素 说 明 如 表 21.5 所 示 。 

















表 21.5 图 书信 息 查询 表单 及 表单 说 明 





属性 设置 

















name="myform" method="post" action="" 












name="txt_book" type="text" id="txt_book" size="25" 








的 结 
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e="submit" name="Submit" value=" 查 询 " 
(2) 建立 与 MySQL 数据 库 的 连接 、 设 置 字符 集 ， 并 返回 数据 库 连接 ID， 代 码 如 下 。 
$link=mysql_connect("localhost","root","root") or die(" 数 据 库 连 接 失 败 ".mysql_error()); // 建 立 与 数据 库 的 连接 


mysql_select_db("db_database21",$link); /| 选择 数据 库 
mysql_query("set names gb2312"); /设置 字符 集 


(3) 应 用 mysql_query0 函 数 执行 SQL 查询 语句 ， 并 使 用 mysql_fetch_object0 函 数 获取 查询 语句 
果 集 ， 代 码 如 下 。 


if ($_POST[Submit==" 查 询 "}{ 
S$txt_book=$_POSTI[txt_book]; // 接 收 查询 关键 字 
$sql=mysql_query("select * from tb_book where bookname like '%".trim($txt_book)."%"); 
// 如 果 选 择 的 条 件 为 “like”， 则 进行 模糊 查询 
$info=mysql_fetch_object($sql); 








} 

(4) 应 用 do…while 循环 语句 以 对 象 的 方式 输出 结果 集中 的 图 书信 息 到 浏览 器 中 ， 代 码 如 下 。 
dof 

?> 


<tr align="left" bgcolor="#FFFFFF"> 
<td height="20" align="center"><?php echo $info->id; ?></td> 
<td >&nbsp;<?php echo $info->bookname; ?></td> 
<td align="center'><?php echo $info->issuDate; ?></td> 
<td align="center"><?php echo $info->price; ?></td> 
<td align="center">&nbsp;<?php echo $info->maker; ?></td> 
<td>&nbsp;<?php echo $info->publisher; ?></td> 

</tr> 

<?php 

}while($info=mysql_fetch_object($sq)l)); 


保存 index.php 动态 页 ， 在 正 浏 览 器 地 址 栏 中 输入 地 址 ， 按 Enter 键 ， 运 行 结果 如 图 21.7 所 示 。 


第 21 章 PHP 管理 MySQL 数据库 中 的 数据 
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图 21.7 查询 图 书信 息 
21.3.6 ”应 用 mysql_fetch_row() 函 数 从 结果 集中 获取 一 行 作为 枚 举 数 组 


mysql_fetch_row0 函 数 从 结果 集中 取得 一 行 作为 枚 举 数 组 ,在 应 用 mysql_fetch_ row0O 函 数 逐 行 获取 
结果 集中 的 记录 时 ， 只 能 使 用 数字 索引 来 读 取 数 组 中 的 数据 ， 其 语法 如 下 。 
array mysql_fetch_row ( resource result ) 


mysql_fetch_rowO 函 数 返回 根据 所 取得 的 行 生成 的 数组 ， 如 果 没 有 更 多 行 则 返回 FALSE。 返 回 数 
组 的 偏 移 量 从 0 开始 ， 即 以 $row[0] 的 形式 访问 第 一 个 元 素 〈 只 有 一 个 元 素 时 也 是 如 此 ) 。 
例 21.6 查询 图 书信 息 ， 并 使 用 mysql_fetch_row0 函 数 获取 结果 集 显 示 图 书信 息 。〔 实 例 位置: 
光盘 \TMNsI21W21.6) 
有 具体 开发 步骤 如 下 。 
(1) 创建 项 目 、 添 加 表单 、 连 接 MySQL 服务 器 以 及 设置 默认 数据 库 的 实现 过 程 与 例 21.5 开发 步 
又 中 的 (1) 、(2) 相同 ， 这 里 不 再 袭 述 。 
(2) 在 应 用 mysql_query0O 函 数 执行 SQL 查询 语句 后 ， 与 例 21.5 不 同 的 是 ， 本 实例 使 用 
mysql_fetch_row0 函 数 获取 查询 语句 的 结果 集 。 代 码 如 下 。 
<?php 
$sql=mysql_query("select * from tb_book"); 
$row=mysql_fetch_row ($sql); 
if ($_POST[Submit]==" 查 询 "){ 
S$txt_book=$_POST[txt_book]; 
// 如 果 选 择 的 条 件 为 “like”， 则 进行 模糊 查询 
S$sql=mysql_query("select * from tb_book where bookname like '%" trim($txt_book)."%"); 
$row=mysql_fetch_row($sql); 
有 


?> 


(3) 应 用 站 条 件 语句 对 结果 集 变 量 $info 进行 判断 ， 如 果 该 值 为 假 ， 则 检索 的 图 书信 息 不 存在 ， 
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应 用 echo 语句 输出 提示 信息 ， 代 码 如 下 。 


<?php 
if($row==falseX{ /| 如果 检索 的 信息 不 存在 ， 则 输出 相应 的 提示 信息 

echo "<div align='center style='color:#FF0000; font-size:12px'> 对 不 起 ， 您 检索 的 图 书信 息 不 存在 !</div>"; 
} 


?> 


(4) 应 用 do…while 循环 语句 以 对 象 的 方式 输出 结果 集中 的 图 书信 息 到 浏览 器 中 ， 代 码 如 下 。 


<?php 

dof 

?> 

<tr align="left" bgcolor="#FFFFFF"> 
<td height="20" align="center"><?php echo $row[0]; ?></td> 
<td >&nbsp;<?php echo $row[1]; ?></td> 
<td align="center"><?php echo $row[2]; ?></td> 
<td align="center"><?php echo $row[3]; ?></td> 
<td align="center">&nbsp;<?php echo $row[4]; ?></td> 
<td>&nbsp;<?php echo $row[5]; ?></td> 

</tr> 

<?php 

}while($row=mysql_fetch_row($sql)); 

?> 


保存 index.php 动态 页 ， 在 正 浏览 器 地 址 栏 中 输入 地 址 ， 按 Enter 键 ， 运 行 结果 如 图 21.8 所 示 。 
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PE 网 络 编程 自学 手册 2012-03-01 明日 科技 | 人 民 邮 电 出 版 社 
PE 程序 开发 范例 宝 奥 2012-02-09 明日 科技 | 人 民 邮电 出 版 社 
IP 数据 库 系统 开发 完全 手册 | 2012-06-12 明日 科技 | 人 民 邮 电 出 版 社 
PIP 函数 参考 大 全 2012-09-18 明日 科技 | 人 民 邮电 出 版 社 
PiP 项 目 开发 全 程 实录 2012-09-24 明日 科技 | 清华 大 学 出 版 社 















































态 本 地 Intranet | 保护 模式 :禁用 各 ” 胞 100% ~ 





图 21.8 ”使 用 mysql_fetch_row0 函 数 获 取 结 果 集 查询 图 书信 息 
21.3.7 ”应 用 mysqlL_num_rows() 函 数 获取 查询 结果 集中 的 记录 数 


使 用 mysql_num rows0 函 数 可 以 获取 由 select 语句 查询 到 的 结果 集中 行 的 数目 ，mysql_ num rowsO 
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函数 的 语法 如 下 。 
int mysqLnum_rows ( resource result ) 
此 命令 仅 对 SELECT 语句 有 效 。 要 取得 被 INSERT、UPDATE 或 者 DELETE 语句 所 影响 到 的 行 的 
数目 ， 要 使 用 mysql_affected rowsO 函 数 。 
例 21.7 使 用 mysql num rows0 函 数 获取 查询 结果 的 记录 数 。〔 实 例 位 置 ， 光 盘 \IMNsN21\21.7) 
具体 开发 步骤 如 下 。 
(1) 建立 与 MySQL 数据 库 的 连接 ， 设 置 字符 集 为 GB2312， 并 返回 数据 库 连 接 ID， 代 码 如 下 。 
$link=mysql_connect("localhost","root","root") or die(" 数 据 库 连 接 失 败 ".mysql_error()); /建立 连接 





mysql_select_db("db_database21",$link); /选择 数据 库 

mysql_query("set names gb2312"); // 设 置 字符 集 
(2) 默认 情况 下 显示 所 有 图 书信 息 ， 代 码 如 下 。 

$sql=mysql_query("select * from tb_book"); // 查 询 所 有 图 书信 息 

$info=mysql_fetch_object($sql); /获取 结果 集 


(3) 如 果 用 户 输入 了 查询 关键 字 ， 并 单 击 了 “查询 ”按钮 ， 则 采用 模糊 查询 的 方式 显示 出 所 有 符 
合 条 件 的 记录 ， 代 码 如 下 。 


if ($_POST[Submit]==" 查 询 "X{ 
S$txt_book=$_POSTTItxt_book]; 
$sql=mysql_query("select * from tb_book where bookname like '%" .trim($txt_book)."%"); 
// 如 果 选 择 的 条 件 为 “like”， 则 进行 模糊 查询 
$info=mysql_fetch_object($sql); 


2 
if($info==falseX{ // 如 果 检 索 的 信息 不 存在 ， 则 输出 相应 的 提示 信息 
echo "<div align='center style='color:#FF0000; font-size:12px'> 对 不 起 ， 您 检索 的 图 书信 息 不 存在 !</div>"; 
} 
dof 
?> 
<tr align="left" bgcolor="#FFFFFF"> 
<td height="20" align="center"><?php echo $info->id; ?></td> 
<td >&nbsp;<?php echo S$info->bookname; ?></td> 
<td align="center"><?php echo $info->issuDate; ?></td> 
<td align="center"><?php echo S$info->price; ?></td> 
<td align="center">&nbsp;<?php echo S$info->maker; ?></td> 
<td>&nbsp;<?php echo $info->publisher; ?></td> 
</tr> 
<?php 
jwhile($info=mysql_fetch_object($sql)); 
?> 


在 正 浏览 器 地 址 栏 中 输入 地 址 ， 按 Enter 键 ， 程 序 默认 输出 图 书信 息 表 中 的 全 部 图 书信 息 ， 并 自 
动 汇总 记录 条 数 ， 如 图 21.9 所 示 。 在 文本 框 中 输入 要 检索 的 图 书 名 称 ， 如 “开发 ” (支持 模糊 查询 ， 
程序 自动 去 除 查询 关键 字 左右 空格 ) ， 单 击 “ 查 询 ” 按 钮 ， 即 可 按 条 件 检索 指定 的 图 书信 息 到 浏览 
并 自动 汇总 检索 到 的 记录 条 数 ， 运 行 结果 如 图 21.10 所 示 。 
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在 默认 页 应 用 mysql_num rows0 忆 [上 寺 应 用 mysql_num rows0 函 数 获 隧 ] 得 8 出 ]| 
函数 获取 数据 表 的 总 记录 数 EE 取 查 询 结果 集中 的 总 记录 数 [Eenr re) 


图 21.9 默认 统计 数据 表 中 所 有 记录 图 21.10 应 用 mysql_num rows0 函 数 获取 查询 结果 集中 的 记录 数 
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[aE 
如 果 要 获取 由 insert、update、delete 语句 所 影响 到 的 数据 行 数 ， 则 必须 应 用 mysql_affected_ 
Tows() 函 数 来 实现 。 


21.3.8 应 用 mysql_free_result() 函 数 释放 内 存 


mysql_free_result0 函 数 用 于 释放 内 存 ， 数 据 库 操作 完成 后 ， 需 要 关闭 结果 集 ， 以 释放 系统 资源 ， 
该 函数 的 语法 如 下 。 

mysql_free_result($result); 

mysql_free_result0 函 数 将 释放 所 有 与 结果 标识 符 result 所 关联 的 内 存 。 该 函数 仅 需要 在 考虑 到 返回 
很 大 的 结果 集 时 会 占用 多 少 内 存 时 调用 。 在 脚本 结束 后 所 有 关联 的 内 存 都 会 被 自动 释放 。 


21.3.9 应 用 mysql_close() 函 数 关闭 连接 


每 使 用 一 次 mysql_connect0 或 mysql_query0 函 数 ， 都 会 消耗 系统 资源 。 在 少量 用 户 访 问 Web 网 站 
时 间 题 还 不 大 ， 但 如 果 用 户 连接 超过 一 定数 量 时 ， 就 会 造成 系统 性 能 的 下 降 ， 甚 至 死机 。 为 了 避免 这 
种 现象 的 发 生 , 在 完成 数据 库 的 操作 后 ， 应 使 用 mysql_close0 函 数 关 闭 与 MySQL 服务 器 的 连接 ， 以 节 
省 系统 资源 。mysql_closeO 函 数 的 语法 如 下 。 
Lclose($conn); 


在 Web 网 站 的 实际 项 目 开 发 过 程 中 ， 经 常 需要 在 Web 页 面 中 查询 数据 信息 。 查 询 后 使 用 
Imysql_closeO 函 数 关闭 数据 源 。 

例 21.8 使 用 mysql_connectO 函 数 建立 与 数据 库 的 连接 ， 然 后 使 用 mysql_select_db0 函 数 选择 数据 
库 并 使 用 mysql_close0 函 数 断 开 与 MySQL 数据 库 的 连接 ， 在 断 开 与 MySQL 数据 库 连 接 后 ， 再 次 使 
mysql_select_db() 函 数 选择 数据 库 ， 从 两 次 选择 数据 库 的 情况 来 判断 mysql_close0 函 数 是 否 能 起 到 断 开 
数据 库 连 接 的 作用 。 实 例 代码 如 下 。〔 实 例 位置 ， 光盘 \TMsl\21\21.8) 


<?php 
$host = "127.0.0.1"; /MySQL 数据 库 服务 器 
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$userName = "root"; /| 用 户 名 

S$password = "root"; /密码 

$dbName = "db_database21"; /| 数据 库 名 

$connlD = mysql_connect($host, $userName, $password); /连接 MySQL 数据 库 
mysql_select_db($dbName, $connID); /| 选择 数据 库 
mysql_close($connlD); // 断 开 与 数据 库 连 接 
mysql_select_db($dbName, $connlD); /| 再 次 选择 数据 库 

x 


运行 本 实例 ， 将 在 页 面 中 输出 如 图 21.11 所 示 的 错误 信息 ， 从 错误 信息 中 可 以 判断 ， 第 一 次 对 数据 
库 选 择 操作 是 成 功 的 ， 而 第 二 次 操作 是 失败 的 ， 即 可 证 明 mysql_close0 函 数 成 功 关 闭 了 与 数据 库 的 连接 。 
(Brep/hocalhosts2/26/268/indexphp - Windo.. [ale sl 
(CON Hav/oclhontz 6+ | x BR 
文件 了 纺 痢 申 查 看 (WV) 收入 天 入 工具 才 助 (H) 
次 Wi 天 ”| 妈 [-[<« | 大 htpy- x 便 - 国 ->” 


























Warning: mysql_select_dbO: 2is not a valid MySQL Link resource 
in E\App Servwnww\26\26 8\indexplp on line 10 
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图 21.11 错误 提示 
21.4 PHP 管理 MySQL 数据 库 中 的 数据 


管理 MySQL 数据 库 中 的 数据 主要 是 对 数据 进行 添加 、 修 改 、 删 除 、 查 询 等 操作 ， 只 有 熟练 地 掌 
握 这 部 分 知识 才能 够 独立 开发 出 基于 PHP 的 数据 库 应 用 。 


21.4.1 向 数据 库 中 添加 数据 


向 数据 库 中 添加 数据 主要 通过 mysql_query0 函 数 和 INSERT 语句 实现 。 
例 21.9 发 表 新 闻 ， 填 写 新 闻 标 题 及 新 闻 内 容 ， 当 用 户 单 击 “提交 ”按钮 时 ， 判 断 新 闻 标 题 及 内 
容 是 否 为 空 ， 如 果 不 为 空 则 将 数据 添加 到 数据 库 中 ， 关 键 代码 如 下 。 实例 位 置 : 光盘 \TMNsI121\21.9) 


<?php 
$conn=mysql_connect("localhost","root","root"); 
mysql_select_db("db_database21",$conn); 
mysql_query("set names uft8"); 
ifisset($_POST[submit]) and $_POST[name"!=null and $_POST[news"!=null and $_POST[submit]==" 提 交 
潍 
S$insert=mysql_query("insert into tb_news(name,news) values(".$_POST[name]",".$_POST[news]."”) 7); 
if($insert)}{ 
echo "<script> alert(' 发 表 成 功 !"); window.location.href='index.php'</script>"; 
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jelse{ 
echo "<script> alert(' 发 表 失 败 !"); window.location.href='index.php'</script>"; 
i 
Jelse{ 
echo "<script> alert(' 发 表 失 败 !"); window.location.href='index.php'</script>"; 
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图 21.12 添加 新 闻 


例 21.10 将 文件 以 二 进 制 的 形式 上 传 到 数据 库 中 ， 并 且 输 出 上 传 的 数据 。 运 行 结果 如 图 21.13 
所 示 。《〔 实 例 位 置 ， 光盘 \TMNsI\21\21.10) 
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图 21.13 将 数据 以 二 进 制 形式 上 传 到 数据 库 
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(1) 通过 Dreamweaver 开发 工具 创建 一 个 index.php 页 ， 添 加 一 个 表单 ， 设 置 表单 的 method 属性 
值 为 post， 设 置 表单 的 action 属性 值 为 ndex_okphp; 添加 表单 元 素 ， 提 交 图 书信 息 ; 添加 “提交 ” 按 
钮 ， 其 关键 代码 如 下 。 
<form name="form" method="post" action="index_ok.php" onSubmit="return check_form(this)"> 
<input name="cover" type="file" id="cover" size="30"> 
<input type="submit" name="Submit" value=" 提 交 "> 
</form> 


(2) 创建 conn 文件 夹 ， 编 写 conn.php 文件， 连接 MySQL 数据 库 服务 器 ， 连 接 db_database21 数 
据 库 ， 设 置 数据 库 变量 编码 格式 为 utf8。 

(3) 创建 index_ok.php 文件 ， 通 过 $_ POST 方法 获取 表单 中 提交 的 数据 ， 并 判断 上 传 图 片 的 格式 
是 否 正确 ， 通 过 fopen0 函 数 和 fread0 函 数 读 取 表单 中 提交 的 图 片 数据 ， 编 写 insert 语句 将 数据 添加 到 
指定 的 数据 表 中 ， 其 关键 代码 如 下 。 











<?php 
include ("conn/conn.php"); // 连 接 数 据 库 
$bookname = $_POST["bookname'"]; 1/ 获取 表单 中 提交 的 数据 


S$price = $_POST["price"]; 

$maker = $_POST["maker"]; 

SissuDate = date("Y-m-d H:i:s"); 

S$publisher = $_POST["publisher"]; 

$synopsis = $_POST["synopsis"]; 

if ($_POST['Submit] == true) { 
S$cover = $_FILES["cover][name]; /获取 表单 中 提交 的 图 片 
S$cover_type = strstr($cover, "."); /| 获取 从 "." 到 最 后 的 字符 
if ($cover_type != ".jpg" && $cover_type != ".gif" && $cover_type != 

".JPG" && $cover_type != ".GIF" && $cover_type != ".bmp" && $cover_type (= 


"BMP") { // 判 断 图 片 的 格式 
echo "<script>alert( 封面 图 片 格式 不 对 ， 请 进行 处 理 后 在 上 传 ! '"); window.location.href='index.php'; 
</script>"; 
}else{ 
S$cover=iconv("utf-8","gb2312",$cover); // 设 置 字符 串 的 编码 格式 


$path = "uploadfiles/".$cover; 
@move_uploaded_file($_FILES["cover"]["tmp_name"], $path); 


S$fp = fopen($path, "rb"); /以 二 进 制 形式 打开 图 片 
S$image = addslashes( 
@fread($fp, filesize($path))); // 读 取 二 进 制 的 数据 

/将 数据 添加 到 指定 的 数据 表 中 


$sql = "insert into tb_book(bookname,price,maker,issuDate,publisher,synopsis,cover)values('$bookname’, 
'$price','$maker','$issuDate','$publisher','$synopsis','$image')"; 
Sresult = mysql_query($sql, 
$conn); 
if ($result == true) { 
echo iconv("utf-8","gb2312"," 文 件 上 传 成 功 !""); 
echo "<meta http-equiv=\"refresh\" content=\"30 url=index.php\">"; 
}else{ 
echo iconv("utf-8","gb2312"," 文 件 上 传 失败 !1"); 
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echo "<meta http-equiv=\"refresh\" content=\"30 url=index.php\">"; 


21.4.2 ”浏览 数据 库 中 的 数据 


浏览 数据 库 中 的 数据 通过 mysql_ query0 函 数 和 select 语句 查询 数据 ,使 用 mysql_fetch_assoc0 函 数 
将 查询 结果 返回 到 数组 中 。 
例 21.11 浏览 tb_news 表 中 的 新 闻 信 息 ， 具 体 代码 如 下 。《 实 例 位 置 : 光盘 \TMNsI21W21.11) 





<?php 
”连接 数据 库 */ 
$conn=mysql_connect("localhost","root","root"); /| 连接 数据 库 服务 器 
mysql_select_db("db_database21",$conn); /| 选择 数据 库 
mysql_query("set names uft8"); // 设 置 编 码 格 式 
$arr=mysql_query("select * from tb_news",$conn); /执行 查询 语句 
/使 用 while 语句 循环 mysql_fetch_assoc() 函 数 返 回 的 数组 */ 
while($result=mysql_fetch_assoc($arr)}{ // 循 环 输出 查询 结果 
i 
<tr> 
<td height="25"><?php echo $resultrname];?> &nbsp;</td> ”<!-- 输 出 新 闻 标题 -> 
<td height="25"><?php echo $resultrnews'];?> </td> <!-- 输 出 新 闻 内 容 --> 
</tr> 
<?php 
) /结束 while 循环 
Fa 
运行 结果 如 图 21.14 所 示 。 
E ee 





新 闻 内 容 
EL PS a 技术 次 
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21.4.3 ”编辑 数据 库 数据 


编辑 数据 主要 通过 mysql_query0 函 数 和 update 语句 实现 。 
例 21.12 ”编辑 新 闻 信 息 表 中 的 新 闻 信 息 ， 有 具体 步骤 如 下 。 (实例 位 置 : 光盘 \TMNsIN21\21.12) 
(1) 创建 数据 库 连 接 文件 connphp， 代 码 如 下 。 


<?php 
$conn=mysql_connect("localhost","root","root"); // 连 接 数 据 库 服 务 器 
mysql_select_db("db_database21",$conn); /连接 db_database21 数据 库 
mysql_query("set names uft8"); // 设 置 数据 库 编码 格式 
人 > 

(2) 创建 ndex.php 文件 ， 显 示 所 有 新 闻 人 信息， 代码 如 下 。 
<?php 
include("conn.php"); // 包 含 conn.php 文 件 
$arr=mysql_query("select * from tb_news",$conn); // 查 询 数 据 


/使 用 while 语句 循环 mysql_fetch_array() 函 数 返 回 的 数组 */ 
while($result=mysql_fetch_array($arr)}{ 


Tr 
<tr> 
<td height="25"><?php echo $resultrname'];?>< 上 -输出 新 闻 标题 -->&nbsp;</td> 
<td><?php echo $resultrnews];?> <!- 输 出 新 闻 内 容 --> </td> 
<td><label> 
<input type="hidden" name="id" value="<?php echo $result[id];?>" /> 
<div align="center"><a href="update.php?id=<?php echo $resultfid1];?>"> 编 辑 </a></div> 
</label></td> 
</tr> 
<?php 
} /结束 while 循环 
?> 


(3) 创建 update.php 文件 ， 显 示 要 编辑 的 新 闻 内 容 ， 代 码 如 下 。 


<form id="form1" name="form1" method="post" action="update_ok.php"> 


<?php 

include("conn.php"); // 包 含 conn.php 文 件 

$arr=mysql_query("select * from tb_news where id=".$_GETTid]."",$conn); /定义 查询 语句 
$select=mysql_fetch_array($arr); // 循 环 输出 查询 内 容 
?> 


<input name="name" type="text" size="40" value="<?php echo $select[name'];?>"/> 
<textarea name="news" cols="40" rows="10"><?php echo $select[news'];?></textarea> 
<input type="submit" name="Submit" value=" 保 存 " /> 
<input type="hidden" name="id" value="<?php echo $select[id];?>" /> 
</form> 
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(4) 创建 update_ok.php 文件 ， 完 成 新 闻 信息 的 编辑 操作 ， 代 码 如 下 。 


<?php 

include("conn.php"); // 包 含 conn.php 文 件 

ifisset($_POST[id]) and isset($_POST[ Submit]) and $_POST[Submit]==" 保 存 "X 
Supdate=mysql_query("update tb_news set name=".$_POST[name]",news=".$_POST[news]."” where 

id=".$_POST[id]."",$conn); 


if($update)\{ 
echo "<script> alert(' 修 改 成 功 !); window.location.href='index.php'</script>"; 
Jelse{ 
echo "<script> alert(' 修 改 失 败 "); window.location.href='index.php'</script>"; 
} 
} 
> 


运行 结果 如 图 21.15 所 示 。 





新 闻 标题 新 闻 内 容 


人 包括 实例 资源 
术 资 源 库 、 方 案 资 源 库 、 界 面 资源 库 、 工 具 资源 





i 羽 
最 全 面 最 系统 诛 吕 寺 


提供 多 样 学 习 模式 ， 无 论 是 初学 者 还 是 开发 者 都 能 找到 
ee :海量 编程 资源 享用 


六 可 i 
最 易学 最 易 用 


a | 通过 不 同 关键 字 ,快速 查询 , 查 您 所 
全 方位 、 人 4 项目 、 方 案 、 技 
要 TREE 立即 应 用 ， 


程序 源码 立即 查询 、 快 速 应 用 ， 加 快 二 次 开发 速度 、 节 
省 时 间 。 用 量 短 的 时 间 开 发 最 好 的 产品 。 


实现 个 人 的 篇 程 丙 源 , 有 效 的 科学 各 理 , 无 论 是 收 计 图 
片 、 代 码 及 其 地 各 种 有 价值 的 内 容 ,都 能 随时 冲 理 随时 
更 新 ， 以 备 日 后 查 、 学 所 用 。 














图 21.15 编辑 新 闻 信 息 


21.4.4 删除 数据 


数据 的 删除 应 用 delete 语句 , 而 在 PHP 中 需要 通过 mysql_query0 函 数 来 执行 这 个 delete 删除 语句 ， 


完成 MySQL 数据 库 中 数据 的 删除 操作 。 
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例 21.13 删除 新 闻 信息 表 中 的 新 闻 信息 ， 具 体 步骤 如 下 。〔 实 例 位 置 : 光盘 \TMNsN21\21.13) 
(1) 创建 数据 库 连 接 文件 conn.php， 代 码 如 下 。 


<?php 

$conn=mysql_connect("localhost","root","root"); /| 连接 数据 库 服务 器 
mysql_select_db("db_database21",$conn); /连接 数据 库 db_database21 
mysql_query("set names uft8"); // 设 置 数据 库 编码 格式 

?> 


(2) 创建 delete php 文件 ， 
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显示 所 有 新 闻 信 息 ， 代 码 如 下 。 


<?php 
include("conn.php"); // 包 含 conn.php 文 件 
$arr=mysql_query("select * from tb_news",$conn); /查询 数据 


/使 用 while 语句 循环 mysql_fetch_array() 函 数 返回 的 数组 */ 
while($result=mysql_fetch_array($arr)){ 


六 
<tr> 
<td height="25"><?php echo $result[name'];?> <!-- 输 出 新 闻 标题 ->&nbsp;</td> 
<td><?php echo $result[news'];?> <!- 输 出 新 闻 内 容 --> </td> 
<td><label> 
<input type="hidden" name="id" value="<?php echo $result[id];?>" /> 
<div align="center"><a href="delete_ok.php?id=<?php echo $result[id1];?>"> 删 除 </a></div> 

</label></td> 
</tr> 

<?php 

} /| 结束 while 循环 
?> 


(3) 在 delete_ okphp 文件 中 


<?php 
include("conn.php"); 


if($delete){ 





， 根 据 超 链接 传递 的 IJD 值 ， 完 成 删除 新 闻 信 息 操 作 ， 代 码 如 下 。 


// 包 含 conn.php 文件 
$delete=mysql_query("delete from tb_news where id=".$_GET[id]."",$conn); /执行 删除 操作 


echo "<script> alert(' 删 除 成 功 !); window.location.href='delete.php'</script>"; 


Jelse{ 


echo "<script> alert(' 删 除 失败 '"); window.location.href='delete.php'</script>"; 


} 


?> 


执行 程序 ， 运 行 结果 如 图 21.16 所 示 。 





攻 








新 闻 标 题 


新 闻 内 容 





编程 局 和 一 一 
是 全 面 最 条 统计。 


蝙 程 词 奥 一 一 


景 昌 学 景 昌 用 “不 忌 . 


编程 二 入 一 一 
高 效 查 询 评 度 需 
检索 

编程 词典 一 一 


开发 

编程 词典 一 一 
个 人 资源 管理 
器 





目前 国内 景 系统 、 景 全 面 的 开发 资源 ,包括 实例 资源 
ee 方案 资源 库 、 界 面 资源 库 、 工 具 资源 


pp 无 沦 是 初学 者 还 是 开发 者 者 能 找到 
最 适合 自己 的 学 习 方式 和 使 用 方式 ,海量 六 程 资源 训 用 


多 种 查询 方式 ， 通 过 不 同 关键 字 ， 块 束 查询 ， 查 您 所 
需 ， 找 您 所 要 。 全 方位 、 评 层次 检索 项目、 方案 、 技 
术 、 实 例 快速 定位 ， 立即 应 用 。 


程序 源码 立即 查询 、 块 速 应 用 ,加 块 二 次 开发 连 度 、 节 
省 时 间 。 用 景 短 的 时 间 开发 景 好 的 产品 


实现 个 人 的 编程 资源 , 有 效 的 科学 管理 ,无论 是 收 意图 
片 、 代 东 及 其 地 各 种 有 价值 的 内 容 ， 乔 能 驮 时 答 理 了 时 
新 ， 以 和 日 后 查 .学 所 用 。 











图 21.16 删除 新 闻 信 息 


353 


MySQL 从 入 门 到 精通 


21.4.5 ”批量 删除 数据 


对 数据 库 中 的 数据 进行 管理 过 程 中 ， 如 果 要 删除 的 数据 非常 多 ， 执 行 单条 删除 数据 的 操作 就 不 适 
合 了 ， 这 时 应 该 使 用 批量 删除 数据 来 实现 数据 库 中 信息 的 删除 。 通 过 数据 的 批量 删除 可 以 快速 删除 多 
条 数据 ， 减 少 操作 执行 的 时 间 。 

例 21.14 ”批量 清理 新 闻 信息 表 中 陈旧 的 新 闻 信息 ， 具 体 步 又 如 下 。【〔 实 例 位 置 ， 光盘 \ 
TMN\sN21\21.14) 

(1) 创建 数据 库 连接 文件 conn.php， 代 码 如 下 。 





<?php 

$conn=mysql_connect("localhost","root","root"); // 连 接 数 据 库 服 务 器 
mysql_select_db("db_database21",$conn); /| 连接 db_database21 数据 库 
mysql_query("set names uft8"); // 设 置 数据 库 编 码 格 式 

人 


(2) 创建 pl_delete.php 文件 ， 显 示 所 有 新 闻 信 息 ， 代 码 如 下 。 


<?php 
include("conn.php"); 
$arr=mysql_query("select * from tb_news",$conn); // 查 询 数 据 
/使 用 while 语句 循环 mysql_fetch_array() 函 数 返回 的 数组 */ 
while($result=mysql_fetch_array($arr)}{ 
yp 
<tr> 
<td><label> 
<label> 
<input type="checkbox" name="checkbox[]" value="<?php echo $result[id];?>" /> 
</label> 
</label></td> 
<td height="25"><?php echo $resultrname];?><!-- 输 出 新 闻 标 题 -></td> 
<td><?php echo $resultrnews1];?><!-- 输 出 新 闻 内 容 --></td> 


<?php 
} // 结 束 while 循环 


(3) 创建 pl_deletel.php 页 面 ， 完 成 批量 删除 操作 ， 代 码 如 下 。 





<?php 
include("conn.php"); // 包 含 conn.php 文件 
if(isset($_POST[Submit]) and $_POST[Submit]==" 删 除 " and $_POST[checkbox"]!="X// 判 断 是 否 执 行 删除 操作 
for($i=0;$i<count($_POST[checkbox]):$i++)f // 遍 历 复 选 框 获取 到 的 新 闻 id 序号 
$sql=mysql_query("delete from tb_news where id=".$_POST[checkbox][$i].",$conn);// 执 行 删除 操作 
} 
if($sql)}{ 
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echo "<script> alert(' 删 除 成 功 !); window.location.href='pl_delete.php'</script>"; 
Jelse{ 
echo "<script> alert(' 删 除 失败 !"); window.location.href='pl_delete.php'</script>"; 
} 
echo "<script> alert(' 请 选择 要 删除 的 内 容 !"); window.location.href='pl_delete.php'</script>"; 
} 
?> 


执行 程序 ， 运 行 结果 如 图 21.17 所 示 。 








编程 词典 一 目前 国内 最 系统 、 最 全 面 的 开发 资源 ， 包括 实 全 资源 库 、 
人 、 方 案 资源 库 、 界 面 资源 库 、 工 具 资 源 库 等 


. 


编程 词典 一 一 提供 多 样 学 习 模式 ， 无 论 是 初学 者 还 是 开发 者 都 能 找到 最 
最 易学 最 易 用 适合 自己 的 学 习 方式 和 使 用 方式 ， 海 量 编程 资源 享用 不 


尽 。 
编程 亲 向 一 名 种 查询 方式 ,通过 不 同 关键 字 ， 快 速 查 疝 ， 查 您 所 
高 重度 找 人 所 要 。 全 方位 、 宁 民 闪 检 索 ， 项 目 、 方 案 、 技 术 、 实 
检索 便 决 束 定位 ， 立即 应 用 

编 各 典 一 和 

: 程 席 源 码 立 妈 坦 询 、 快速 让 用， 加 快 二 次 开发 这 度 、 节 省 
尖 守 用 加 快 时 间 。 用 最 适时 间 开 发 最 好 的 产品 。 








21.17 ”批量 删除 数据 信息 


21.5 PHP 操作 MySQL 事务 


事务 处 理 机 制 在 程序 开发 过 程 中 有 着 非常 重要 的 作用 ， 它 可 以 使 整个 系统 更 加 安全 。 例 如 ， 在 银 
行 处 理 转账 业务 时 ， 如 果 A 账户 中 的 金额 刚 被 发 出 ， 而 B 账户 还 没 来 得 及 接收 就 发 生 停电 ， 会 给 银行 
和 个 人 带 来 很 大 的 经 济 损失 。 采 用 事务 处 理 机 制 ， 一 旦 在 转账 过 程 中 发 生意 外 ， 则 程序 将 回 滚 ， 不 做 
任何 处 理 。 

下 面 讲解 使 用 事务 处 理 技术 实现 关联 表 信息 的 删除 方法 。 

例 21.15 采用 事务 处 理 方式 ， 对 学 生 信息 表 和 学 生成 绩 表 中 的 数据 进行 删除 。 运 行 本 实例 ， 学 
生 信 息 表 及 学 生成 绩 表 信 息 分 别 如 图 21.18 和 图 21.19 所 示 。 当 删除 图 21.18 中 的 学 生 信息 后 ， 查 看 
学 生成 绩 信息 时 可 以 发 现 ， 与 该 学 生 对 应 的 成 绩 也 被 全 部 删除 。 代 码 如 下 。〔 实 例 位 置 ， 光盘 
\IMNsI21W21.15) 

















FE 


站 


到 
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学 生成 绩 管 理 系统 


查看 学 生 和 信息 





住址 身份 证 号 
0312317 吉林 省 长 春 市 22010245++** 
0312316 吉林 省 吉林 市 2021103285S+errt 

















21.18 查看 学 生 信息 


学 生成 绩 管理 系统 





学 号 
0312317 
0312316 
0312316 
0312317 



































21.19 查看 学 生成 绩 


程序 开发 步骤 如 下 。 
(1) 建立 数据 库 及 数据 表 并 实现 与 数据 的 连接 ， 代 码 如 下 。 


<?php 

$conn=new mysqli("localhost","root","root","db_database21"); 
$conn->query("set names utf8"); 

?> 


(2) 显示 所 有 学 生 的 基本 信息 ， 代 码 如 下 。 


<?php 
include_once("conn.php"); 
$sql=$conn->query("select * from tb_stu"); 
S$info=$sql->fetch_array(MYSQLI_ASSOC); 
if($info==NULL) 


{ 
echo " 暂 无 学 生 信息 ""; 


<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo S$info[sname];?></div></td> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo S$info[sno];?></div></td> 

<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo S$info[sage]:?></div></td> 

<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo S$info[saddress];?></div></td> 
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<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo S$info[ssfzh];?></div></td> 
<td bgcolor="#FFFFFF"><div align="center"><a href="javascript:if(window.confirm(' 确 定 删除 该 学 生 信 
息 么 ?')==true){window.location.href='delete.php?id=<?php echo $info[id];?>";}"> 删 除 </a></div></td> 


</tr> 
<?php 


while($info=$sql->fetch_array(MYSQLI_ASSOC)); 


} 


?> 


在 实现 该 模块 功能 时 ， 可 以 利用 JavaScript 实现 该 页 与 delete.php 页 面 的 信息 传递 ， 代 码 如 下 。 
<a href="javascript:if(window.confirm(' 确 定 删除 该 学 生 信息 路 ')==true) 


{window.location . 


</a> 


href='delete.php?id=<?php echo $info[id];?>";}"> 删 除 


window 对 象 的 confirm0 方 法 用 于 弹出 一 个 对 话 框 ， 提 示 用 户 真正 删除 某 学 生 的 所 有 信息 。 
(3) 利用 事务 处 理 机 制 对 学 生 的 基本 信息 进行 删除 ， 代 码 如 下 。 


<?php 
$id=$_GET[id]; 


include_once("conn.php"); 
$conn->autocommit(false); 
if(!$conn->query("delete from tb_sco where id=".$id."")) 


$conn->rollback(); 


if(I$conn->query("delete from tb_stu where id=".$id."")) 


$conn->rollback(); 


$conn->commit(); 
$conn->autocommit(true); 
echo "<script>window.location.href="index.php';</script>"; 


?> 


MySQL 5.0 以 后 的 版 本 开始 支持 存储 过 程 ， 存 储 过 程 具 有 一 臻 性、 高效 性 、 安 全 性 
特点 ， 本 节 将 具体 讲解 PHP 是 如 何 操 纵 MySQL 存储 过 程 的 。 
下 面 介绍 如 何 使 用 存储 过 程 实现 用 户 登录 。 


21.6 PHP 操作 MySQL 存储 过 程 








例 21.16 使 


中 登录 窗口 的 文本 框 中 输入 用 户 名 和 密码 ， 然 后 单 击 “ 登 录 ” 按 钮 ， 如 果 用 户 登 录 的 





确 , 则 会 将 页 面 定向 





MySQL 存储 过 程 实现 用 户 登 录 身 份 的 验证 。 运 行 本 实例 ， 如 图 21. 


和 体系 结构 等 


20 所 示 。 在 图 






































户 名 和 密码 正 


到 如 图 21.21 所 示 的 登录 成 功 提示 页 面 。 代 码 如 下 。( 实 例 位 置 : 光 盘 \TMsI\21\21.16) 
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明 自 77 条 闻 网 











Se) https/ocalhostis2/26/26.16/success.php “| 全 | X| 
文件 (P)。 编 纺 (E) ”查看 (V) ” 收 茂 赤 (A) 工具 (D) 帮助 (H) 
| 误 wii 夫 。 红 |- < 乱 使 用 .xx 全 ~- 国 - 口 











医 喜 您 mz， 您 已 经 成 功 登录 本 站 ! 








多 本 地 Intranet | 保护 模式 :禁用 种 ”> 拆 100% ~ 


21.21 登录 成 功 的 提示 信息 


程序 开发 步骤 如 下 。 
(1) 创建 存储 过 程 ， 通 过 query 方法 调用 存储 过 程 ， 代 码 如 下 。 


delimiter // 

create procedure pro_login1(in un varchar(50), in pass varchar(50)) 
begin 

select * from tb_login where username=un and password=pass; 
end 

ll 


(2) 通过 MySQLi 扩展 库 建 立 与 MySQL 数据 库 的 连接 , 并 设置 数据 库 字 符 集 为 utf-8, 代码 如 下 。 


<?php 

$conn=new mysqli("localhost","root","root","db_database21"); 
$conn->query("set names utf8"); 

?> 


(3) 建立 用 户 登录 表单 ， 当 用 户 在 表单 中 录入 用 户 名 和 密码 ， 并 单 击 “ 提 交 ” 按 钮 后 ， 通 过 如 下 
代码 验证 用 户 的 登录 信息 是 否 正确 。 


<?php 
iflisset($_POST[username]) && trim($_POST[username'])!=") 
{ 
require_once 'Db.php'; 
$username =trim($_POST[username]); 
S$password = trim($_POST[password]); 
$sql = $mysqli->query("call pro_login1(".$username.", ".$password.™")"); 
Sinfo = $sql->fetch_array(MYSQLIL ASSOC): 
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if($info = null}{ 
$_SESSION[IoginUsermname] = $username; 
echo '<script>window.location.href="success.php";</script>"; 
jelse{ 
echo '<div style="width:300px; height:30px; line-height:30px; border:1px solid #E59B04; 
background-color:#FCF2E0; color:#FF0000;"> 用 户 名 或 密码 输入 有 误 </div>'; 
> 
} 


?> 


上 述 代码 中 ， 首 先 判断 $_POST[username'] 的 值 是 否 被 设置 ， 则 首先 使 用 require_once 语句 包含 
conn.php 文件 ， 然 后 使 用 MySQLi 扩展 库 的 query0 方 法 执行 存储 过 程 pro_ login， 并 使 用 query0 函 数 返 
可 的 对 象 调用 fetch_array0 函 数 获得 结果 集 ， 最 后 通过 判断 结果 集 是 否 为 null 来 判断 用 户 所 录入 的 信 
息 是 否 正确 。 














21.7 常见 问题 与 解决 方法 


在 实际 PHP 与 MySQL 结合 的 应 用 中 ， 往 往 会 碰 到 很 多 问题 。 这 些 问题 不 一 定 是 由 于 代码 的 错误 
造成 的 ， 可 能 是 一 些 环境 因素 ， 或 者 是 数据 存在 的 问题 。 本 节 将 介绍 一 些 常见 问题 及 相应 的 解决 方案 。 


21.7.1 MySQL 服务 器 无 法 连接 


背 误 信息 如 下 所 示 。 


Warning:mysql_connect() [function.mysql-connect]: Unkown MySQL server host 'ocalhost(11001) in 
E:\wamp\wwwtest.php on line 2 

出 现 这 条 错误 信息 的 原因 可 能 有 以 下 几 点 。 

(1) 代码 中 的 mysql_connect 函数 中 指定 的 服务 器 地 址 有 误 。 

(2) 数据 库 服务 器 不 可 用 。 

解决 方案 如 下 。 

(1) 检查 代码 中 的 服务 器 地 址 是 否 正 确 。 

(2) 检查 数据 库 服 务 器 是 否 已 经 启动 并 且 可 用 。 





21.7.2 ”用户 无 权限 访问 MySQL 服务 器 
错误 信息 如 下 。 


Warning:mysqlL_connect() [function.mysql-connect]: Access denied for user "root@ ‘localhost'(using 
password:NO) in E:\Wwamp\www\test.php on line 2 
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出 现 这 条 错误 信息 的 原因 可 能 是 代码 中 的 mysql_connect 函数 中 能 够 指定 的 用 户 名 或 者 密码 有 误 
或 者 在 当前 服务 器 上 不 可 用 。 

解决 方案 如 下 。 

检查 代码 中 的 用 户 名 和 密码 是 否 正 确 。 

通过 MySQL 命令 行 测试 是 否 可 以 使 用 该 用 户 名 和 密码 登录 MySQL 数据 库 服务 器 。 





21.7.3 ”提示 mysql_connect 函数 未 定义 


错误 信息 如 下 。 

Fatal error : Callto undefined function mysql_connect() in E:\wamp\wwwtest.php on line 2 

出 现 这 条 错误 信息 的 原因 可 能 是 在 php.ini 文件 中 没有 配置 MySQL 的 扩展 库 。 

解决 方案 如 下 。 

编辑 php.ini 文件 ， 定 位 到 如 下 位 置 ， 去 掉 此 项 前 面 的 分 号 ， 保 存 后 重新 启动 Apache 服务 器 。 
;extension=php_mysql.dll 





21.7.4 SQL 语句 出 错 或 没有 返回 正确 的 结果 


这 种 情况 经 常 在 使 用 动态 SQL 语句 时 出 现 ， 以 下 代码 就 存在 一 个 错误 。 


<?php 
mysql_connect("localhost", "root", "root") or die; 
mysql_select_db("db_database21 "); 
$sql="select * from $table"; 
$result=mysql_query($sql); 
print_r(mysql_fetch_row($result)); 

Ca 


上 述 代 码 中 错误 地 使 用 了 一 个 没有 赋值 的 变量 $table 作为 操作 的 数据 表 名 称 , 结果 返回 如 下 错误 








Warning: mysqlLfetch_row(): supplied argument is not a valid MySQL result resource in 

E:wamp\wwwindex.php on line 6 

解决 方案 : 使 用 print 或 者 echo 函数 输出 SQL 语句 来 检查 错误 。 例 如 ， 对 上 述 代码 进行 修改 ， 通 
过 echo 语句 直接 输出 $sql 的 值 ， 查 看 这 个 SQL 语句 是 否 正确 ， 其 代码 如 下 。 


<?php 
mysql_connect("localhost", "root", "root") or die; /| 连接 数据 库 
mysql_select_db("db_database21 "); /| 选择 数据 库 
$sql="select * from S$table"; /定义 SQL 语句 
echo $sql; /| 输出 SQL 语句 
Sresult=mysql_query($sql); /执行 SQL 语句 
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print_r(mysql_fetch_row($result)); 1// 输 出 执行 结果 


?> 


如 此 ， 从 运行 结果 就 可 以 看 出 SQL 语句 中 的 错误 了 。 


21.7.5 ”数据库 乱码 问题 





在 获取 数据 库 中 的 数据 时 ， 中 文字 符 串 的 输出 出 现 乱 码 。 

问题 分 析 : 输出 数据 库 中 的 数据 之 所 以 会 出 现 乱 码 ， 是 因为 在 获取 数据 库 中 的 数据 时 ， 数 据 本 身 
所 使 用 的 编码 格式 与 当前 页 面 的 编码 格式 不 符 ， 从 而 导致 输出 数据 乱码 。 

解决 方案 : 在 与 MySQL 服务 器 和 指定 数据 库 建 立 连 接 后 ， 应 用 mysql_query0 函 数 设置 数据 库 中 
字符 的 编码 格式 ， 使 其 与 页 面 中 的 编码 格式 一 致 。 





<?php 

$conn=mysql_connect("localhost","root","root"); // 连 接 数 据 库 服务 器 
mysql_select_db("db_database21",$conn); // 连 接 数据 库 db_database21 
mysql_query("set names uft8"); // 设 置 数据 库 编码 格式 

?> 


上 述 通 过 mysql_ query0O 函 数 设 置 的 编码 格式 是 ut8， 同 样 还 可 以 设置 其 他 编码 格式 ， 唯 一 的 一 个 
条 件 就 是 要 与 数据 库 中 的 编码 格式 相 匹配 。 

这 就 是 解决 数据 库 中 中 文 输出 乱码 的 方法 , 应 用 mysql_query 函数 设置 数据 库 的 编码 格式 , 使 其 与 
页 面 中 编码 格式 保持 一 致 ， 也 就 不 会 出 现 乱 码 的 问题 。 


21.7.6 ”应 用 MYSQL_ERROR() 语 句 输出 错误 信息 


在 执行 MySQL 语句 时 产生 的 错误 是 很 难 发 现 的 ， 因 为 在 PHP 脚本 中 执行 一 个 MySQL 的 添加 、 
查询 、 删 除 语句 时 ， 如 果 是 MySQL 语句 本 身 的 错误 ， 程 序 中 不 会 输出 任何 的 信息 ， 除 非 对 MySQL 语 
句 的 执行 进行 判断 ， 成 功 输 出 什么 ， 失 败 输出 什么 。 

解决 方案 : 为 了 查找 出 MySQL 语句 执行 中 的 错误 ， 可 以 通过 mysql_errorO 语 句 来 对 SQL 语句 进 
行 判断 ， 如 果 存 在 错误 则 返回 错误 信息 ， 和 否则 没有 输出 ， 该 语句 的 应 用 被 放置 于 mysql_query0 函 数 
之 后 。 

例如 ， 在 下 面 的 代码 中 ， 在 通过 mysql query0 函 数 执行 查询 语句 之 后 ， 应 用 mysql_error0 函 数 获 
取 SQL 语句 中 的 错误 。 


<?php 





$sql="select * from tb_new"; /定义 查询 语句 
$query=mysql_query($sql,$conn); /执行 查询 操作 
echo mysql_error(); /获取 SQL 语句 中 的 错误 
while($myrow=mysql_fetch_array($query)){ /循环 输出 查询 结果 

?> 
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此 方法 不 仅 对 查询 语句 的 执行 有 效 ， 而 且 对 添加 、 更 新 和 删除 语句 都 适用 ， 是 一 个 查找 SQL 语句 
本 身 错误 的 好 方法 。 


21.8 小 结 


本 章 首 先 对 PHP 语言 进行 简单 的 介绍 , 让 读者 先 对 PHP 语言 有 一 个 基本 的 了 解 , 然后 介绍 了 PHP 
访问 MySQL 数据 库 的 一 般 流 程 ， 并 且 详细 介绍 了 该 流程 每 一 步骤 的 具体 实现 方法 ， 接 下 来 再 通过 实 
例 更 深层 次 地 介绍 了 PHP 如 何 实现 对 MySQL 数据 库 进行 增 、 删 、 改 、 查 操作 , 最 后 介绍 了 如 何在 PHP 
中 操作 事务 和 存储 过 程 ， 以 及 PHP 管理 MySQL 数据 库 时 经 常 遇 到 的 问题 的 解决 方法 。 通 过 本 章 的 学 
习 ， 读 者 能 够 掌握 PHP 操作 MySQL 数据 库 的 一 般 流程 ， 掌 握 常 用 MySQL 函数 的 使 用 方法 ， 并 能 够 
具备 独立 完成 基本 数据 库 程 序 的 能 力 。 希 望 本 章 能 够 起 到 抛砖引玉 的 作用 ， 能 够 帮助 读者 在 此 基础 上 
更 深层 次 地 学 习 PHP 操作 MySQL 数据 库 的 相关 技术 , 并 进一步 学 习 使 用 面向 对 象 的 方式 操作 MySQL 
数据 库 的 方法 。 


21.9 ”实践 与 练习 


1. 编写 一 个 实例 ， 实 现 以 分 页 的 方式 显示 员工 信息 ， 并 通过 输入 页 码 跳 转 到 指定 的 页 。 (答案 位 
置 ， 光盘 \IMNsM2121.17) 

2. 编写 一 个 实例 ， 实 现 查 询 图 书信 息 表 中 的 前 3 条 记录 。〔 答 案 位 置 ， 光盘 \TMNsI21\21.18) 

3. 编写 一 个 实例 ， 实 现 对 图 书信 息 表 中 的 图 书信 息 按 图 书 序号 进行 降序 排列 。 (答案 位 置 ， 光盘 
\IM\sI21W21.19) 
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让 第 23 章 Struts 2+Spring+Hibernate+MySQL 实现 网 络 商城 


本 篇 分 别 使 用 PHP 和 Java 两 种 语言 ， 结 合 My5QL 实现 了 两 个 大 型 的 、 完 整 
的 管理 系统 ， 通 过 这 两 个 项 目 ， 运 用 软件 工程 的 设计 思想 ， 让 读者 学 习 如 何 进行 软 
件 项 目的 实践 开发 。 书 中 按照 编写 系统 分 析 一 系统 设计 一 数据 库 与 教 据 表 设 计 一 公 
共 模 块 设计 一 创建 项 目 一 实现 项 目 一 项 目 和 总 结 的 过 程 进 行 介绍 ,带领 读者 一 步 一 步 
亲身 体验 开发 项 目的 全 过 程 。 
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所 谓 网 上 社区 是 指 包 括 BB% 论 坛 、 聊 天 室 、 博 客 等 形式 在 内 的 网 上 交流 室 间 ， 
同一 主题 的 网 上 社区 集中 了 具有 共同 兴趣 的 访问 者 ， 由 于 有 众多 用 户 的 参与 ， 因 此 
具备 了 交流 的 功能 ， 成 为 一 个 营销 场所 。 网 上 社区 有 各 种 不 同 的 表现 形式 和 规模 
有 个 人 创办 的 社区 ， 功 能 和 界面 追求 时 尚 、 个 性 突出 ; 有 大 型 的 商业 性 质 社 区 ， 以 
盈利 为 目的 ， 分 类 多 元 化 ， 适 合 不 同类 型 的 网 民 。 

本 章 开 发 的 BCTY365 网 上 社区 主要 面向 程序 开发 人 员 ， 集 论坛 、 留 言 板 、 软 
件 下 载 、 升 级 下 载 、 技 术 支持 和 在 线 购物 等 功能 于 一 身 ， 既 是 一 个 程序 开发 者 交流 
的 平台 ， 更 是 一 个 网 络 营销 的 场所 。 

通过 阅读 本 章 ， 读 者 可 以 : 

了 解 如 何 做 需求 分 析 和 系统 设计 

掌握 在 Linux 操作 系统 下 搭建 PHP 开发 环境 
掌握 如 何 设计 和 创建 数据 库 、 数 据 表 

掌握 如 何 设计 公共 模块 

掌握 在 线 支付 技术 

掌握 在 线 论坛 功能 的 实现 方法 

掌握 软件 上 传 和 下 载 功 能 的 实现 方法 
掌握 在 Linux 操作 系统 下 发 布 网 站 的 方法 


豆 于 于 于 于 于 于 至 
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22.1 开发 背景 


随 着 市 场 竞 争 的 日 益 激烈 ， 企 业 的 生存 和 发 展 之 路 更 加 艰难 。 要 想 使 企业 保持 旺盛 的 生命 力 ， 企 
业 必 须要 跟 上 时 代 发 展 的 脚步 ， 不 断 为 企业 注入 新 的 活力 。 某 科技 公司 为 适应 市 场 的 需求 ， 增 加 公司 
在 互联 网 上 的 影响 力 ， 将 开发 一 个 网 上 社区 系统 ， 为 广大 的 编程 爱好 者 提供 一 个 交流 的 平台 ， 并 且 以 
此 来 推广 该 公司 的 软件 产品 。 


22.2 系统 分 析 


当 一 个 开发 项 目 被 确立 时 ， 首 先 要 做 的 就 是 需求 分 析 、 可 行 性 分 析 ， 然 后 编写 项 目 计划 书 ， 以 使 
项 目 开 发 人 员 了 解 和 掌握 网 站 的 前 期 策划 和 网 站 开发 流程 。 


22.2.1 需求 分 析 


在 开发 网 上 社区 之 前 ， 首 先 要 明确 所 要 开发 的 社区 属于 什么 类 型 ， 是 个 人 的 社区 系统 ， 还 是 商业 
化 的 社区 系统 ; 并且 要 知道 开发 的 社区 是 面向 什么 样 的 人 群 ， 是 普通 网 民 ， 还 是 专业 的 技术 人 员 ， 或 
者 是 其 他 的 特殊 群体 。 针 对 不 同 的 人 群 ， 社 区 应 该 具有 不 同 的 特点 。 当 明确 了 这 些 ， 项 目 开 发 的 思路 
就 清晰 了 ， 然 后 再 对 网 站 上 一 些 相关 的 社区 进行 考察 、 分 析 ， 从 中 吸取 经 验 ， 并 结合 企业 的 要 求 以 及 
实际 的 市 场 调查 结果 ， 提 出 一 个 合理 的 网 上 社区 网 站 功能 架构 。 本 网 站 需求 如 下 。 

(1) 网 站 设计 页 面 要 求 整洁 、 美 观 大 方 ， 能 够 展示 企业 形象 。 

(2) 网 站 页 面具 有 banner 广告 ， 树 立 企 业 良 好 的 口碑 宣传 。 

(3) 设计 主要 从 编程 者 的 角度 考虑 ， 为 编程 者 解决 在 开发 中 出 现 的 问题 。 

(4) 展示 出 企业 全 力 推 出 的 软件 产品 和 提供 的 免费 软件 ， 以 此 吸引 浏览 者 。 

(5) 提供 一 个 良好 的 网 上 购物 的 操作 平台 。 

(6) 提供 技术 支持 ， 解 决 编程 过 程 中 常见 的 问题 。 

(7) 提供 一 个 讨论 和 研究 问题 的 平台 。 

(8) 做 到 让 广大 浏览 者 关注 企业 的 动态 。 

(9) 为 客户 提供 反馈 信息 的 平台 ， 能 够 做 到 及 时 与 客户 进行 沟通 。 

(10) 完善 的 后 台 管 理 系统 。 











22.2.2 ”可 行 性 分 析 


可 行 性 分 析 是 世界 上 普遍 采用 的 一 种 研究 工程 项 目 是 否 可 行 的 科学 。 其 通过 各 种 有 效 的 方法 ， 对 
工程 项 目 进行 分 析 ， 从 技术 、 经 济 、 市 场 等 方面 加 以 评价 ， 最 终 给 投资 决策 者 提供 是 否 选择 该 项 目 进 
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行 开 发 的 依据 。 
BCTY365 网 上 社区 项 目 开发 的 可 行 性 分 析 主 要 从 以 下 两 个 方面 考虑 。 


1. 经 济 可 行 性 分 析 

企业 为 扩大 公司 的 影响 力 ， 推 出 软件 产品 ， 采 用 网 上 社区 的 形式 在 网 络 上 进行 推广 ， 不 但 可 以 汇 
聚 更 多 的 人 气 ， 而 且 可 以 让 更 多 的 人 了 解 该 企业 ， 从 而 达到 推广 企业 软件 产品 的 目的 ， 最 终 为 企业 带 
来 更 大 的 收益 。 更 重要 的 一 点 是 采取 该 方法 的 成 本 相对 其 他 的 电视 广告 或 者 人 力 宣传 的 成 本 要 低 得 多 ， 
虽然 周期 很 长 ， 但 是 却 能 够 取得 长 期 的 收益 。 

2. 技术 可 行 性 分 析 

网 上 社区 系统 的 开发 采用 的 是 Apache+PHP+phpMyadmin+MySQL 5.6， 开 发 软件 都 是 免费 的 ， 直 
接 可 以 从 网 上 下 载 ， 无 须 支 付 任何 费用 。 要 完成 BCTY365 网 上 社区 系统 的 开发 ， 必 须 能 够 配置 PHP 
程序 开发 的 环境 ， 掌 握 在 线 支付 、 购 物 车 和 在 线 论坛 技术 。 





22.3 系统 设计 


22.3.1 系统 目标 


根据 对 目前 网 络 上 各 种 社区 的 分 析 和 研究 ， 结 合 本 项 目的 实际 需求 ， 在 设计 时 应 该 达到 以 下 目标 。 
(1) 界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵 活 ， 树 立 企业 形象 。 

(2) 功能 完善 、 结 构 清晰 。 

(3) 重点 突出 企业 的 软件 产品 。 

(4) 及 时 更 新 网 站 公告 。 

(5) 及 时 查阅 和 回复 客户 反馈 信息 。 

(6) 为 用 户 提供 沟通 和 交流 的 平台 。 

(7) 购物 车 模块 的 设计 结构 合理 、 流 程 清晰 。 

(8) 购物 结算 功能 设计 符合 逻辑 ， 计 算 准 确 。 

(9) 订单 处 理 功能 的 设计 及 时 、 准 确 、 安 全 。 

(10) 网 上 支付 功能 的 设计 处 理 好 与 网 上 银行 之 问 数据 的 传递 ， 确 保安 全 、 可 靠 。 
(11) 具备 完善 的 后 台 管 理 功能 ， 能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 
(12) 系统 运行 稳定 ， 具 备 良好 的 防范 措施 。 





22.3.2 ”系统 功能 结构 


结合 需求 分 析 和 系统 目标 中 的 内 容 ，BCTY365 网 上 社区 系统 的 功能 结构 已 经 设计 完成 。 为 了 使 读 
者 能 够 更 清楚 地 了 解 网 站 的 结构 ， 下 面 给 出 BCTY365 网 上 社区 前 台 和 后 台 功 能 模块 结构 图 。 
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BCTY365 网 上 社区 前 台 管 理 系统 的 功能 设计 如 图 22.1 所 示 。 
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图 22.1 网 上 社区 前 台 功能 模块 结构 图 
BCTY365 网 上 社区 后 台 管 理 系 统 的 功能 设计 如 图 22.2 所 示 。 
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图 22.2 网 上 社区 后 台 功能 模块 结构 图 
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22.3.3 ”系统 预览 


BCTY365 网 上 社区 系统 由 多 个 程序 页 面 组 成 ， 下 面 给 出 几 个 典型 页 面 ， 其 他 页 面 参 见 光盘 中 的 源 
程序 。 
前 台 首页 如 图 22.3 所 示 ， 该 页 面 用 于 展示 本 系统 的 功能 模块 ， 突 出 企业 的 形象 ， 推 广 企业 的 软件 
品 。 后 台 首 页 如 图 22.4 所 示 ， 该 页 面 用 于 实现 对 编程 词典 、 技 术 支 持 、 软 件 升 级 、 软 件 试用 等 内 容 
的 管理 。 


CS ectY365 社 区 网 ， 找 你 只 要 的 
4 一 oe ee er 





ee 
Fes 
Suna | 
ET 





ET 





mS 
| PBS ON: G01- criean om 启用 朋 HI 有 有 有 
图 22.3 前 台 首页 (光盘 \TM\sI\22Vbcty36S\index.php) 22.4 后 台 首页 (光盘 \TM\sl\22\bcty365\admin\default.php) 
在 线 订 购 模块 的 页 面 效果 如 图 22.5 所 示 ， 该 页 面 主要 用 于 展示 本 企业 在 线 推出 的 软件 产品 ， 实 现 


对 产品 的 在 线 购买 功能 。 软 件 下 载 模块 的 页 面 效 果 如 图 22.6 所 示 ， 该 页 面 主要 用 于 展示 本 企业 提供 的 
免费 软件 ， 并 且 提供 下 载 链 接 。 


区 网 ， 找 你 想 要 的 


0 





图 22.5 在 线 订购 (光盘 \TM\s122\bcty365\ morebccd.php) 图 22.6 软件 下 载 (光盘 \TM\s122\bety365\rjxz.php) 


社区 论坛 模块 的 页 面 效 果 如 图 22.7 所 示 ， 该 页 面 主要 用 于 展示 论坛 中 的 各 大 版 块 ， 并 且 提 供 超 链 
接 跳 转 到 对 应 的 版 块 。 后 台 登 录 页 面 效果 如 图 22.8 所 示 ， 该 页 面 主要 实现 后 台 管 理 员 登 录 。 


过 
be 
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图 22.7 社区 论坛 (光盘 \TMNsl22\bcty365\bbs index.php) 图 22.8 ”后台 登录 (光盘 \TM\sl\22\bcty365\admin\index.php) 


22.3.4 


在 开发 


开发 环境 


BCTY365 网 上 社区 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 


1. 服务 器 端 
操作 系统 : Windows 7/Linux (推荐 ) 。 


服务 器 : 


Apache 2.2.11。 


PHP 软件 : PHP 5.2.6。 


数据 库 : 


MySQL 


MySQL 5.6.20。 
图 形 化 管理 软件 : PhpMyAdmin-4.2.8。 


开发 工具 : Dreamweaver 8。 


浏览 器 : 
分 辨 率 : 


IE 6.0 及 以 上 版 本 。 
最 佳 效 果 1024X768 像素 。 


2. 客户 端 


浏览 器 : 
分 辩 率 : 


22.3.5 


在 进行 
的 分 类 ， 分 
条 理 清晰 ， 


正 6.0 及 以 上 版 本 。 
最 佳 效 果 1024X768 像素 。 


文件 夹 组 织 结构 
网 站 开发 之 前 ， 要 对 网 站 的 整体 文件 夹 组 织 架构 进行 规划 。 对 网 站 中 使 用 的 文件 进行 合理 


别 放 置 于 不 同 的 文件 夹 下 。 通 过 对 文件 夹 组 织 架构 的 规划 ， 可 以 确保 网 站 文件 目录 明确 、 
同样 也 便于 网 站 后 期 的 更 新 和 维护 。 本 案例 的 文件 夹 组 织 架构 规划 如 图 22.9 所 示 。 
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用 于 存 镶 网 站 后 台 管理 文件 

用 于 存储 编程 词典 的 界面 

用 于 存储 网 站 后 台 使 用 的 css 文件 
用 于 存储 网 站 后 台 的 图 片 文件 


用 于 存储 升级 下 载 的 文件 
用 于 存储 可 下 载 的 软件 
用 于 存储 连接 数据 库 的 文件 


用 于 存储 网 站 前 台 使 用 的 css 文件 
用 于 存储 网 站 前 台 使 用 的 图 片 文件 
用 于 存储 网 站 论坛 中 上 传 的 图 片 文件 








图 22.9 文件 夹 组 织 结构 


22.4 在 Linux 操作 系统 下 搭建 PHP 开发 环境 


Red Hat Linux 9 是 Linux 众多 版 本 中 比较 大 众 化 的 一 版 。 在 安装 系统 时 ， 如 果 选 择 完全 安装 或 者 选 
择 Apache、MySQL、PHP 的 安装 包 ， 则 三 者 将 被 安装 到 系统 中 ， 用 户 只 需 将 Apache 和 MySQL 服务 启 
动 就 可 以 使 用 二 者 ， 非 常 方便 ， 但 是 Apache 和 MySQL 的 版 本 可 能 不 是 很 理想 。 为 了 能 够 创建 一 个 良好 
的 PHP 开发 环境 ， 这 里 将 详细 介绍 自行 在 Linux 下 安装 和 配置 Apache 2+MySQL 5.0+PHP 5 的 方法 。 

首先 应 该 到 相关 官方 网 站 下 载 三 者 的 安装 包 。 

(1) httpd-2.0.58.tar.gz 或 更 高 版 本 (http:/httpd.apache.org/) 。 

(2) mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 或 更 高 版 本 (http:/www-.oracle.conyindex.html) 。 

(3) php-5.0.0.tar.gz 或 更 高 版 本 http://www.php.net/》。 

libxml2-2.5.10.tar.gz 或 更 高 版 本 (如 果 读 者 系统 中 的 libxml2 的 版 本 已 经 等 于 或 高 于 该 版 本 可 以 不 
必 下 载 该 安装 包 ) 。 


22.4.1 Linux 下 Apache 的 安装 配置 


首先 将 下 载 的 httpd 安装 包 复制 到 适当 的 位 置 ， 例 如 /usr/locaVwork 下 《如果 目录 不 存在 ， 可 以 建 
立 该 目录 ) 。 打 开 Red Hat Linux 9 的 主 菜单 ， 选 择 “ 系 统 工具 ”， 在 弹出 的 菜单 中 选择 “终端 ”命令 ， 
将 打开 如 图 22.10 所 示 的 终端 窗口 。 


~ WT x 
| 文 作 思 护 狂 E) 查看 (V) 终 绸 D 转 球 G) 帮助 H) 
[roote@loca lhost roo!]# 








22.10 ”Red Hat Linux 9 的 终端 命令 窗口 


a 
| 
Linux 下 Apache、MySQL 及 PHP 的 安装 都 是 在 如 图 22.10 所 示 的 终端 命令 窗口 通过 命令 方式 
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在 该 窗口 中 输入 如 下 命令 进入 work 目录 。 

cd /usr/local/work 

在 work 目录 中 输入 如 下 命令 解压 httpd-2.0.58.tar.gz。 
tar xfz httpd-2.0.58.tar.gz 

进入 解压 后 的 目录 httpd-2.0.58。 

cd httpd-2.0.58 

建立 makefile， 并 将 Apache2 安装 到 /usr/local/apache2 目录 下 。 
-configure -prefix=/usrlocalapache -enable-module=so 
开始 编译 。 

make 

开始 安装 到 设置 的 目录 去 。 

make install 


至 此 ，Apache 2 的 安装 工作 完成 ， 可 以 在 每 次 启动 系统 时 通过 如 下 命令 启动 或 重新 启动 Apache 2 
服务 。 


/usrlocalapache2/bin/apachectl start 
/usr/local/lapache2/bin/apachectl restart 


打开 浏览 器 , 在 地 址 栏 中 输入 http://127.0.0.1 或 者 http://localhost, 按 Enter 键 , 如 果 出 现 如 图 22.11 
所 示 的 页 面 ， 则 说 明 Apache 2 安装 成 功 。 



































E23 = SI 
文件 (E》 编辑 (E) ”查看 (WD 半 到 (G) 书 答 B) 工具 ID。 窗口 WD 帮助 (了 
总 -这 她 、 区 [| hep//ocaboien php -2 续 ~ 
各 主页 | 只 书签 和 Red Hat Network 萝 Suppor 欧 Shop 项 Pmducs 项 Taining 
您 能 看 见 这 个 页 面 ， ache web_server 已 经 安装 成 功 。 您 可 以 在 这 个 目录 中 增加 内 二 
容 ， 或 者 把 这 个 页 面 
这 不 是 你 想 看 见 的 页 面 吧 ? 


Fe he 是 因为 网 站 管理 员 改 变 了 这 个 站 点 的 设置 。 如 胖 有 疑问 ， 请 咨询 维 
护 此 站 点 的 人 员 。 Apache 软件 基金 会 ， 即 此 站 点 所 使 用 的 网 站 服务 器 软件 的 开发 者 ， 不 负 
re 也 无 法 为 您 解决 设置 上 的 问题 。 


Apache 文档 已 经 包含 在 此 发 行 版 中 。 
您 可 以 在 使 用 Apache 的 网 站 服务 器 上 ， 自 由 地 使 用 下 面 的 图 片 。 感 谢 使 用 Apachel 


ss 一 一 一 二 一 


4] J > 
泥 瑟 也 国 For 


必 色 人生 去 画 生起 


图 22.11 测试 Apache 服务 器 
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22.4.2 Linux 下 MySQL 的 安装 配置 
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将 mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 复制 到 /usr/local/work 目录 下 ， 建 立 MySQL 账号 。 
groupadd mysql 

在 组 群 中 加 入 MySQL。 

useradd -g mysql mysql 

进入 local 目录 。 

cd /usr/local 

将 mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 解压 到 该 目录 。 

tar xfz /usr/local/work/mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 

考虑 到 MySQL 数据 库 升 级 的 需要 ， 所 以 通常 以 链接 的 方式 建立 asrlocalmysql 目录 。 
In -s mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz mysql 

进入 MySQL 目录 。 

cd mysql 

在 /usr/local/mysql/data 中 建立 MySQL 的 数据 库 。 

scripts/mysql_install_db -user=mysql 

修改 文件 权限 。 


chown -R root . 
chown -R mysql data 
chgrp -R mysql . 


到 此 MySQL 安装 成 功 。 可 以 通过 在 终端 中 输入 如 下 命令 启动 MySQL 服务 。 
/usr/local/mysql/bin/mysqld_safe -user=mysql& 

启动 MySQL 后 输入 如 下 命令 查看 安装 结果 。 

/usrlocalmysqlybin/mysql -uroot 

如 果 终 端 窗口 中 出 现 如 图 22.12 所 示 的 提示 ， 则 说 明 MySQL 安装 成 功 。 





文件 E) ”编辑 (E) ”查看 (VW) 次 端 (D 转 到 G) 帮助 HHD 

[root@loca lhost root]# /usr/local/nysql/bin/nysql -uroot -p123 
Wlcone to the MSQL nonitor. Conmnds end with ;or \g. 

Your MSQL connection id is 1 to server version: 5.0.0-alpha-standard 


Type ‘help:’ or ‘\h’ for help. Type ‘\e’ to clear the buffer 


mysql> 目 





图 22.12 测试 MySQL 
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22.4.3 Linux 下 PHP 的 安装 配置 


首先 查看 系统 中 libxml2 的 版 本 号 ， 如 果 libxml2 的 版 本 号 小 于 2.5.10， 则 需要 安装 libxml2-2.5.10. 
tar.gz 或 更 高 版 本 ， 因 为 PHP 5 必须 在 libxml2 的 版 本 大 于 2.5.10 的 前 提 下 才能 够 安装 。 
将 libxml2-2.5.10.tar.gz 复制 到 /usrlocal/work 目录 下 ， 并 进入 该 目录 。 


cd /usr/local/work 





解压 libxml2-2.5.10.tar.gz。 

tar xfz libxml2-2.5.10.tar.gz 

进入 该 目录 。 

cd libxml2-2.10 

建立 makefie 并 将 libxml2 安装 到 /usrlocallibxml2 下 。 


.configure —prefix=/usr/local/libxml2 


开始 编译 。 

make 

开始 安装 到 设置 的 目录 去 。 
make install 


至 此 libxml2 安装 成 功 。 

将 php-5.0.0.tar.gz 复制 到 /usr/local/work 目录 下 ， 并 进入 该 目录 。 
cd /usr/local/work 

解压 php-5.0.0.tar.gz。 

tar xfz php-5.0.0.tar.gz 

进入 php-5.0.0 目录 。 

cd php-5.0.0 


建立 makefile。 

./configure —with-apxs2=/usr/local/lapache2/bin/apxs \ 
-With-mysql=/usr/local/mysql \ 
—With-libxml-dir=/usr/local/libxml2 


开始 编译 。 
make 
开始 安装 。 


make install 


373 


MySQL 从 入 门 到 精通 


复制 php.ini-dist 或 php.ini-recommended 到 /usrlocallib 目录 ， 并 命名 为 php.ini。 

cp php.ini-dist /usr/local/lib/php.ini 

更 改 httpd.conf 文件 相关 设置 , 该 文件 位 于 /usr/local/apache2/conf 中 。 找到 该 文件 中 的 如 下 指令 行 。 
AddType application/x-gzip .gz .tgz 

在 该 指令 后 添加 如 下 指令 。 

AddType application/x-httpd-php .php .phtml 

和 新 启动 Apache， 并 在 Apache 主 目 录 下 建立 文件 test.php。 





<?php 
phpinfo(); 


?> 


在 浏览 器 地 址 栏 中 输入 http://127.0.0.1/test.php， 按 Enter 键 ， 如 果 出 现 如 图 22.13 所 示 的 页 面 ， 则 
PHP 安装 成 功 。 
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和 
后 过 ” 前进 ” 重新 载 入 
| 侣 主页 | 叭 书签 帮 Red Hat Network 旷 Suppon 莉 Shop 的 Products 手 Trining 
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22.13 PHP 测试 


Apache2 默认 主 目录 为 /usrlocal/apache2/htdocs。 


安装 文件 的 路 径 要 遵循 一 定 的 客观 原则 ， 为 了 避免 在 Windows 和 Linux 间 移 植 程序 时 带 来 的 不 
， 便 ,选择 D:\usr\local\php 的 目录 时 要 和 在 Linux 下 的 安装 目录 相 匹配 。 建 议 最 好 不 要 选择 中 Rae 
的 目录 ， 如 E:\program FilesPHP， 人 演 发 生 。 
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22.5 ”数据库 设计 


开发 一 个 功能 完善 的 网 上 社区 离 不 开 数 据 库 的 支持 ， 只 有 拥有 了 强大 的 数据 库 ， 网 上 社区 才能 够 


存储 大 量 的 数据 信息 ， 实 现 更 多 、 更 好 的 功能 来 吸引 更 多 的 社 
据 库 的 设计 进行 详细 介绍 。 


22.5.1 数据 库 分 析 


BCTY365 网 上 社区 是 一 个 中 型 的 面向 软件 开发 者 的 程序 ， 








区 成 员 。 本 节 将 对 BCTY365 网 上 社区 数 


考虑 到 开发 的 成 本 、 搭 配 的 合理 性 以 及 


操作 的 灵活 性 等 ， 使 用 了 MySQL 数据 库 。 从 成 本 考虑 MySQL 数据 库 是 完全 免费 的 ， 可 以 在 网 上 免费 
下 载 ; 从 匹配 的 角度 讲 PHP 与 MySQL 数据 库 一 直 是 公认 的 最 佳 搭档 ; MySQL 数据 库 不 但 可 以 在 命令 


模式 下 进行 操作 ， 而 且 还 配备 了 一 些 比较 流行 的 图 形 化 管理 工 


MySQL 数据 库 中 的 数据 进行 操作 。 
22.5.2 ”数据 库 概 念 设计 


有 具 ， 如 phpMyAdmin 等 ， 可 以 轻松 地 对 


根据 上 述 各 节 对 BCTY365 网 上 社区 系统 做 的 需求 分 析 和 系统 设计 ， 规 划 出 BCTY365 网 上 社区 的 
实体 关系 E-R 图 。 其 中 包括 注册 用 户 实体 、 发 帖 信息 实体 、 回 帖 信息 实体 、 订 单 信息 实体 和 编程 词典 
信息 实体 ， 其 他 还 有 一 些 辅助 的 实体 ， 是 对 上 述 实 体 中 内 容 的 补充 。 由 于 涉及 的 实体 较 多 ， 这 里 只 对 
注册 用 户 实体 、 发 帖 信息 实体 和 订单 信息 实体 的 E-R 图 进行 介绍 。 


1. 注册 用 户 实体 








注册 用 户 实体 用 于 存储 用 户 注册 信息 ， 包 括 编号 、 用 户 名 、 真 实 姓名 、 密 码 、 邮 箱 、 性 别 、 电 话 、 
QQ 号 码 、 家 庭 地 址 、 访 问 次 数 、 注 册 时 间 、 最 后 一 次 登录 时 间 、 卫 地 址 、 邮 政 编码 、 用 户 类 型 、 密 


码 提示 问题 、 密 码 答案 、 真 实 密码 、 表 情 图 、 发 帖 次 数 属性 。 


注册 用 户 实 体 的 E-R 图 如 图 22.14 所 示 。 








图 22.14 注册 用 户 实体 E-R 图 
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2. 发 帖 信息 实体 

发 帖 信息 实体 用 于 存储 登录 本 社区 的 会 员 在 论坛 中 发 布 帖子 的 相关 信息 ， 包 括 编 号 、 用 户 名 ID、 
帖子 类 别 、 帖 子 标题 、 帖 子 内 容 、 发 帖 时 间 、 最 后 回复 时 间 、 表 情 图 、 访 问 次 数 、 是 否 项 帖 、 上 传 图 
片 属性 。 发 帖 信息 实体 的 E-R 图 如 图 22.15 所 示 。 





[发帖 信息 ] 





图 22.15 发 帖 信息 实体 的 E-R 图 
3. 订单 信息 实体 
订单 信息 实体 存储 用 户 在 线 购 买 时 填写 的 订单 信息 ， 包 括 编号 、 用 户 名 、 性 别 、 家 庭 地 址 、 邮 政 
编码 、QQ 号 码 、 邮 箱 、 手 机 号 码 、 电 话 号 码 、 收 货 方式 、 邮 资 、 产 品 金 额 、 订 单 时 间 、 订 单 号 、 选 择 
城市 等 属性 。 订 单 信息 实体 的 E-R 图 如 图 22.16 所 示 。 








订单 信息 





22.16 订单 信息 实体 E-R 
22.5.3 创建 数据 库 及 数据 表 


在 BCTY365 网 上 社区 系统 中 应 用 的 是 db_bcty365 数据 库 ， 其 中 涉及 18 个 数据 表 ， 数 据 表 的 名 称 
和 功能 如 图 22.17 所 示 。 
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加 服务 器 : localhost ， 昌 数据 库 : db_bcty365 

表 类 型 说 明 
th_bb MyISAM ”gb2312_chinese_ci ”编程 词典 版 本 信息 表 
th_bbqb MylSAM ”gb2312_chinese_ci 版 本 之 间 区 别 信息 表 
th_bbs MylSAM ”gb2312_chinese_ci ”论坛 发 帖 信 息 表 
th_bccd MylSAM ”gb2312_chinese_ci 编程 词典 信息 表 
th_city MylSAM gb2312_chinese_ci “城市 信息 表 
th_cjwt MyISAM gb2312_chinese_ci 常见 问题 信息 表 
tb_dd MyISAM gb2312_chinese_ci 订单 信息 表 
th_reply MyISAM gb2312_chinese_ci ”论坛 回帖 信息 表 
th_sjxz MylSAM ”gb2312_chinese_ci 软件 升级 下 载 信 息 表 
th_soft MylSAM gb2312_chinese_ci 软件 下 载 信息 表 
th_tell MylSAM gb2312_chinese_ci 。 社区 公告 信息 表 
th_type MylSAM ”gb2312_chinese_ci ”社区 模块 类 型 信息 表 
也 type_big MylSAM gb2312_chinese_ci 论坛 大 类 信息 表 
tb_type_small MylSAM gb2312_chinese_ci 论坛 小 类 信息 表 
th_user MyISAM ”gb2312_chinese_ci “注册 用 户 信息 表 
th_xIh MylISAM ”gb2312_chinese_ci 升级 下 载 序 列 号 信息 表 
th_bccdj MylSAM gb2312_chinese_ci 编程 词典 简介 信息 表 
tb_leaveword MylSAM gb2312_chinese_ci 存储 客户 反馈 信息 








22.17 db_bcty365 数据 库 中 使 用 的 数据 表 


本 案例 中 创建 数据 库 和 数据 表 使 用 的 是 pppMyAdmin 图 形 化 管理 工具 。 下 面 将 介绍 数据 库 和 数据 
表 的 创建 方法 ， 以 及 在 创建 过 程 中 需要 注意 的 一 些 问题 。 


1. 数据 库 的 创建 


打开 phpMyAdmin 图 形 化 管理 工具 的 主页 ， 首 先 在 文本 框 中 输入 要 创建 的 数据 库 的 名 称 〈 如 
db bcty365) ， 然 后 在 下 拉 列 表 框 中 选择 要 使 用 的 字符 编码 格式 ， 这 里 使 用 的 是 gb2312_chinese_ ci， 如 
图 22.18 所 示 。 最 后 单 击 “ 创 建 ”按钮 ， 数 据 库 创建 成 功 。 


localhost phpMyAdmin - 2.9.0.2 

—]QL client version: 4.1.7 

， Protocol verkion: 10 H PHP extensions: mysql 

困 服 务 器 :localhost via TCPIP 字 Language 国 : [中 文 - Chinese simplified 司 


名 全 轩 傈 phpMyAdmin 官方 网 站 
(db beya6s | * IChangeLogl [CVSI Lists 


((9b2312_chinese_ci 
国字 再 宁可 


选择 编码 格式 


本 Import phpM yAdmin 


























22.18” phpMyAdmin 管理 界面 
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,| 区 六 万 
创建 数据 库 的 过 程 中 ,尽量 使 用 与 程序 内 容 贴切 的 英文 字符 进行 命名 ,有 助 于 对 数据 库 的 理解 ; 
如 果 使 用 AppServ 配置 PHP 开发 环境 那么 在 创建 数据 库 时 不 需要 指定 编码 的 格式 ， 默 认 值 为 
gb2312 chinese ci; 如 果 自 行 配置 开发 环境 那么 就 要 指定 编码 格式 为 gb2312_chinese_ ci， 否则 创 
建 数据 库 的 编码 格式 为 latin1 swedish_ ci， 将 导致 数据 库 中 数据 出 现 乱 码 。 


2. 创建 数据 表 


在 成 功 创建 数据 库 以 后 ， 接 下 来 就 是 创建 数据 表 ， 这 里 以 tb_bb 编程 词典 版 本 信息 表 为 例 ， 讲 解 
如 何 创建 数据 表 ， 以 及 在 创建 数据 表 的 过 程 中 都 需要 注意 哪些 问题 。 这 里 创建 一 个 名 称 为 tb_bb 的 数 
据 表 ， 包 括 3 个 字段 ， 如 图 22.19 所 示 。 














甸 在 数据 库 db_bcy365 中 创建 一 个 新 表 
名 蕊 |b bb ) Numberoffiel 人 JJ3 _) 


| sss | ser | 


图 22.19 创建 tb_bb 数据 表 
单 击 “ 执 行 ”按钮 后 ， 进 入 到 如 图 22.20 所 示 的 添加 字段 信息 的 页 面 中 , 在 此 处 对 字段 进行 详细 的 
设置 ， 包 括 字 段 名 、 数 据 类 型 、 长 度 / 值 、 属 性 、 默 认 值 、 额 外 、 主 键 和 索引 等 。 


胃 服务 器 : localhost ， 轧 数据 库 : db_bcty365 ， 国 表 :tb_bb 
字 及 & 类 型 四 长 度 / 值 *1 Null 默 W2 ”| 设置 主键 图 
id INT 了 10 not null 加 nt 可 al 
rc 





| 
bbname VARCHAR 本 |200 not null 加 了 
[createtime DATETIME 可 notnull 机 





指定 编码 格式 
指定 数据 表 的 类 型 hE @ 
C [mas _¥| I Mab232 chinesecl 


图 22.20 添加 数据 表 中 字段 信息 


,| 区 攻 万 
创建 数据 库 中 的 数据 表 时 , 字段 名 的 设计 尽量 要 与 数据 表 的 内 容 相符 合 , 这 样 有 助 于 程序 后 期 
维护 和 修改 工作 的 进行 ， 能 够 直观 地 看 出 数据 表 的 作用 。 

如 果 使 用 AppServ 配置 PHP 开发 环境 ， 那 么 在 创建 数据 表 时 不 需要 指定 数据 表 类 型 和 编码 格 
式 ; 如 果 使 用 自行 配置 的 PHP 开发 环境 ， 那 么 就 要 指定 数据 表 的 类 型 为 MyISAM 和 字符 的 编码 格 
式 为 gb2312_chinese ci; 否则 创建 的 数据 表 类 型 为 hnoDB， 而 编码 格式 为 latinl swedish_ ci， 将 导 

致 该 数据 表 中 的 数据 复制 到 其 他 机 器 上 不 可 用 ， 并 且 数 据 表 中 的 数据 出 现 乱 码 。 
在 创建 数据 表 的 过 程 中 ， 一 定 要 为 数据 表 指 定 一 个 主键 ， 它 是 数据 表 的 一 个 唯一 的 标识 。 





掌握 数据 表 的 创建 方法 后 ， 就 可 以 自行 创建 数据 表 。 由 于 本 案例 中 涉及 的 数据 表 有 17 个 之 多 ， 这 
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里 不 能 对 每 个 数据 表 的 功能 设计 进行 一 一 介绍 ， 所 以 只 给 出 几 个 重要 的 数据 表 的 设计 效果 图 供 广 大 读 
者 参考 ， 其 他 数据 表 请 参见 本 书 附带 的 光盘 。 数 据 表 的 设计 结构 如 图 22.21~ 图 22.23 所 示 。 
1) tb_user (注册 用 户 信息 表 ) 
注册 用 户 信息 表 主 要 用 于 存储 本 社区 中 会 员 的 个 人 信息 。 该 数据 表 的 结构 如 图 22.21 所 示 。 
胃 服务 器 : localhost ， 电 数据 库 : db_bcty365 ， 国 表 : tb_user 




















字段 类 型 整理 性 性 Null 默认 额外 说 明 
过 int(8) 否 auto_increment 。 自动 编号 ID 
usernc varchar(50) gb2312_chinese_ci 是 NULL 注册 用 户 名 
truename varchar(50) gb2312_chinese_ci 是 NULL 真实 姓名 
pwd varchar(50) gb2312_chinese_ci 是 NULL 注册 密码 
email varchar(50) gb2312_chinese_ci 是 NULL 有 效 邮箱 地 址 
Sex varchar(2) gb2312_chinese_ci 是 NULL 性 别 
tel varchar(20) 。 gb2312_chinese_ci 是 NULL 联系 电话 
qq varchar(20) gb2312_chinese_ci 是 NULL 晤 号 码 
address varchar(100) gb2312_chinese_ci 是 WULL 联系 地 址 
logintimes cint(8) 否 访问 次 数 
regtime datetime 否 注册 时 间 
lastlogintime ”datetime 否 最 后 一 次 登录 时 间 
ip varchar(20) gb2312_chinese_ci 否 I? 地 址 
吊 varchar(20) gb2312_chinese_ci 是 NULL 邮政 编码 
usertype int(2) 理 用 户 类 型 
guestion Yarchar(200) gb2312_chinese_ci 否 密码 提示 问题 
answer varchar(200) gb2312_chinese_ci 否 密码 提示 答案 
truepwd varchar(200) gb2312_chinese_ci 否 真实 密码 
photo varchar(50) ”gb2312_chinese_ci 否 表情 图 
pubtimes int(4) 是 0 发 帖 次 数 


图 22.21 注册 用 户 信息 表 结构 
2) tb_ reply (论坛 回帖 信息 表 ) 




















论坛 回帖 信息 表 主 要 用 于 存储 登录 会 员 在 本 社区 中 回复 帖子 的 信息 。 该 数据 表 的 结构 如 图 22.22 

加 服务 器 : localhost ， 电 数据 库 : db_bcty365 》 表 :tb_reply 

字段 类 型 整理 尾 性 Null 默认 额外 说 明 
记 int(8) 否 auto_increment ”自动 编号 ID 
userid int(9) 否 0 注册 用 户 ID 
bbsid int(8) L.A] 发 布 帖子 表 ID 
title varchar(200) gb2312_chinese_ci 是 AULL 回复 主题 
content mediumtext gb2312_chinese_ci 是 NULL 回复 内 容 
createtime datetime 是 WULL 回复 时 间 
mark int(2) 是 NULL 回复 记录 
photo varchar(80) 。 gb2312_chinese_ci 是 NULL 回复 图 片 


图 22.22 ”论坛 回帖 信息 表 结构 
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3) tb_bccd (编程 词典 信息 表 ) 
编程 词典 信息 表 主 要 用 于 存储 本 社区 的 在 线 订 购 模块 中 出 售 的 编程 词典 的 基本 信息 。 该 数据 表 的 
结构 如 图 22.23 所 示 。 
胃 服务 器 : localhost 总 数据 库 : db_bcty365 ， 国 表 :tb_bccd 


宇 及 类 型 整理 尾 性 Null 默认 额外 说 明 
过 int(8) 否 auto_increment 。 自动 编号 ID 
bccdname varchar(200) gb2312_chinese_ci 否 编程 词典 名 称 
owner varchar(100) gb2312_chinese_ci 否 开发 者 
typeid varchar(50) gb2312_chinese_ci 否 版 本 类 型 
content mediumtext 。 gb2312_chinese_ci 否 编程 词典 简介 
samepart mediumtext 。 gb2312_chinese_ci 再 软件 共同 点 
addtime datetime 理 开发 时 间 
imageaddress varchar(100) gyb2312_chinese_ci 否 界面 存储 地 址 
bbid int(8) 是 NULL 所 属 版 本 的 ID 
price float 是 NULL 价格 


图 22.23 ”编程 词典 信息 表 结 构 
22.6 公共 模块 设计 


22.6.1 数据 库 连接 文件 


在 进行 程序 开发 的 过 程 中 ， 有 很 多 地 方 都 涉及 数据 库 的 应 用 ， 在 应 用 数据 库 之 前 首先 要 与 数据 库 
建立 连接 ， 因 此 可 以 将 数据 库 的 连接 代码 作为 一 个 公共 文件 进行 存储 ， 在 需要 使 用 数据 库 连 接 文 件 的 
地 方 直接 调用 该 文件 即 可 。 无 须 重复 编写 相同 的 代码 ， 既 减少 了 代码 的 元 余 ， 也 便于 对 数据 库 连 接 文 
件 进 行 修改 。 在 本 项 目 中 笔者 将 数据 库 的 连接 代码 存储 于 conn.php 中 。conn.php 文件 的 代码 如 下 。 

代码 位 置 光盘 \TM\sl\22\bcty365\conn\conn.php 





<?php 
$conn=mysql_connect("localhost","root","root"); /| 连接 数据 库 服务 器 
mysql_select_db("db_bcty365",$conn); // 连 接 指定 的 数据 库 


mysql_query("set names gb2312"); ”// 指 定数 据 库 中 字符 的 编码 格式 为 gb2312， 避 免 出 现 中 文 乱码 的 问题 
?> 
成 功 创建 conn.php 文件 后 , 在 需要 进行 数据 库 的 操作 程序 中 , 就 可 以 通过 include 或 者 其 他 包含 语 
句 调 用 conn.php 该 文件 即 可 ， 无 须 再 编写 连接 数据 库 的 程序 代码 。 应 用 include 语句 包含 conn.php 文 
件 的 代码 如 下 。 
<?php 
include ("conn/conn.php"); // 包 含 数 据 库 文件 


和 
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22.6.2 ”将 文本 中 的 字符 转换 为 HTML 标识 符 


在 输出 数据 库 中 数据 的 过 程 中 ， 有 必要 将 数据 中 的 一 些 特殊 字符 转换 为 HIML 标识 符 ， 这 样 可 以 





避免 一 些 麻烦 。 例 如 ， 在 输出 一 个 程序 的 执行 代码 的 过 程 中 ， 如 果 不 对 其 进行 转换 ， 那 么 输出 的 将 不 
是 程序 的 代码 ,而 是 程序 的 执行 结果 。 这 里 将 文本 中 字符 的 转换 编写 到 一 个 自 定义 函数 unhtml0 中 , 保 
存 到 function.php 文件 中 ， 将 其 作为 一 个 公共 模块 来 使 用 ， 当 需要 使 用 时 直接 调用 function.php 文件 即 
可 。function.php 文件 包含 两 个 自 定义 函数 ， 即 unhtml0 函 数 和 msubstr0 函 数 ，unhtml0 函 数 用 于 将 数据 
中 的 特殊 字符 转换 为 HTML 标识 符 ，msubstr0 函 数 用 于 对 字符 串 进行 指定 长 度 的 截取 。 其 代码 如 下 。 





代码 位 置 : 光盘 \TM\sl22\bcty365\function.php 


<?php 
function unhtml($content)f /定义 自 定义 函数 的 名 称 
$content=htmlspecialchars($content); /| 转换 文本 中 的 特殊 字符 
$content=str_replace(chr(13),"<br>",$content); // 革 换文 本 中 的 换行 符 
$content=str_replace(chr(32),"&nbsp;",$content); // 营 换文 本 中 的 &nbsp; 
S$content=str_replace("[L[","<",$content]; /车 换文 本 中 的 大 于 号 
S$content=str_replace(")_)",">",$content); /| 替换 文本 中 的 小 于 号 
$content=str_replace("|_|"," ",$content); /车 换文 本 中 的 空格 
return trim($content); /删除 文本 中 首尾 的 空格 


} 
/定义 一 个 用 于 截取 一 段 字符 串 的 函数 msubstr() 
function msubstr($str,$start,$len){ ”//$str 指 的 是 字符 串 ，$start 指 的 是 字符 串 的 起 始 位 置 ，$len 指 的 是 长 度 
S$strlen=$start+ $len; // 用 $strlen 存储 字符 串 的 总 长 度 〈 从 字符 串 的 起 始 位 置 到 字符 串 的 总 长 度 ) 
for($i=0;$i<$strlen, $i++){ // 通 过 for 循环 语句 ， 循 环 读 取 字符 串 
if(ord(substr($str,$i,1))>0xa0X{ // 如 果 字 符 串 中 首 个 字 节 的 ASCII 序数 值 大 于 0xa0， 则 表示 为 汉字 
S$tmpstr.=substr($srt,$i,2); // 每 次 取出 两 位 字符 赋 给 变量 $tmpstr， 即 等 于 一 个 汉字 


S$i++i; /变量 自 加 1 
Jelse{ // 如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 $tmpstr 
S$tmpstr.=substr($str,$i,1);} 
} 
return $tmpstr; // 输 出 字符 串 
}> 


22.7 前 台 首 页 设计 


当今 时 代 ， 人 们 都 十 分 重视 对 事物 的 第 一 印象 ， 第 一 印象 基本 上 就 决定 了 对 某 个 事物 的 看 法 和 态 





度 ， 在 网 络 中 更 是 如 此 ， 网 站 给 人 的 第 一 印象 如 果 不 好 ， 那 么 就 会 有 很 多 人 因此 而 不 去 浏览 该 网 站 ， 
无 论 网 站 的 内 容 是 否 丰 富 。 可 以 说 网 站 首页 设计 的 成 功 与 否 直接 影响 着 整个 网 站 的 发 展 。 
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22.7.1 前 台 首 页 概述 


网 站 首页 是 整个 网 站 的 脸面 ， 既 要 突出 企业 的 形象 ， 又 要 展示 出 网 站 强大 的 功能 。 如 果 网 站 首页 
设计 得 非常 成 功 ， 那 无 疑 是 为 整个 网 站 的 成 功 增添 了 一 个 硅 码 。BCTY365 网 上 社区 首页 的 设计 以 企业 
的 品牌 形象 为 基础 ， 全 力 打造 网 站 的 整体 功能 ， 重 点 推出 企业 的 软件 产品 ， 有 具体 内 容 如 下 。 

(1) 网 站 菜单 导航 : 首页 、 技 术 支 持 、 在 线 订购 、 社 区 论坛 、 软 件 下 载 、 升 级 下 载 、 购 买 须知 、 
联系 我 们 。 

(2) 用 户 注册 和 登录 模块 实现 用 户 注册 、 会 员 登录 、 找 回 密码 和 修改 密码 的 功能 。 

(3) 网 站 公告 : 主要 用 于 发 布 社区 中 的 一 些 新 消息 和 重大 事件 。 

(4) 编程 词典 模块 : 推广 企业 的 软件 产品 。 

(5) 软件 下 载 模块 : 展示 企业 提供 的 适用 版 和 免费 的 软件 产品 。 

(6) 常见 问题 模块 : 列举 出 编程 中 常见 问题 的 解决 方案 。 

(7) 社区 论坛 模块 : 浏览 社区 论坛 中 的 部 分 帖子 。 

(8) 升级 下 载 模块 : 提供 一 些 软件 的 升级 版 本 下 载 。 

上 述 内 容 就 是 BCTY365 网 上 社区 首页 中 体现 出 的 内 容 ， 为 了 更 加 直观 地 了 解 网 上 社区 首页 的 设计 ， 
这 里 先 预览 一 下 社区 首页 ， 该 首页 在 本 书 光盘 中 的 路 径 为 \TM\02\bcty365\index.php， 如 图 22.24 所 示 。 
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图 22.24 BCTY365 网 上 社区 首页 
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BCTY365 网 上 社区 首页 的 设计 看 上 去 很 复杂 ， 由 很 多 个 版 块 组 成 ， 但 实现 的 过 程 非常 简单 。 总 体 
架构 使 用 一 个 两 行 三 列表 格 和 一 个 三 行 三 列 的 表格 ， 将 其 分 隔 成 不 同 的 版 块 ， 然 后 使 用 脚本 语句 从 数 
据 库 中 读 取 数 据 ， 最 后 将 数据 循环 输出 到 页 面 中 ， 其 中 网 站 的 头 尾 文 件 使 用 include 包含 语句 调用 。 首 
页 的 框架 结构 如 图 22.25 所 示 。 
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22.25 ”网 站 首页 的 框架 结构 


22.7.2 前 台 首 页 技术 分 析 


作为 网 站 首页 ， 不 一 定 要 具有 什么 特殊 的 技术 或 者 功能 ， 应 该 是 以 简洁 、 鲜 明 ， 突 出 企业 形象 ， 
展示 网 站 的 功能 为 主 。 即 使 使 用 的 是 一 个 静态 页 面 ， 只 要 能 够 将 内 容 表达 全 面 、 完 整 ， 那 么 这 个 首页 
设计 也 是 非常 成 功 的 。 

在 本 案例 首页 的 设计 中 ， 应 用 到 一 个 文字 循环 滚动 的 技术 ， 通 过 该 技术 来 输出 社区 中 发 布 的 公告 
信息 。 该 技术 是 通过 JavaScript 脚本 和 div 标签 来 共同 实现 的 ， 其 实现 的 原理 是 : 首先 创建 一 个 div 标 
签 ， 然 后 在 div 标签 中 输出 公告 信息 ， 最 后 通过 JavaScript 来 对 div 标签 进行 操作 ， 实 现 div 标签 中 内 
容 的 滚动 输出 。 该 技术 的 实现 在 index.php 页 中 完成 ， 其 中 使 用 的 JavaScript 脚本 的 代码 如 下 。 

代码 位 置 ， 光盘 \TMsl\22\bcty36S\index.php 

<script language="JavaScript"> 


marqueesHeight=222; /定义 输出 标签 的 高 度 
stopscroll=false; /定义 stopscroll 的 默认 值 为 false 
with(marquees\{ /| 编辑 marquees 标签 的 属性 
style.width=0; /定义 初始 宽 为 0 
style.height=marqueesHeight; /定义 marquees 标签 的 高 为 222 
style.overflowX="visible"; /定义 值 为 显示 
style.overflowY="hidden"; // 定 义 值 为 隐藏 
noWrap=true; 
‘onmouseover=new Function("stopscroll=true"); // 当 鼠标 经 过 时 执行 stopscroll=true 


MySQL 从 入 门 到 精通 
‘onmouseout=new Function("stopscroll=false"); // 当 鼠标 离开 时 执行 stopscroll=false 


} 
// 创 建 一 个 新 的 div"templayer" 与 div"marquees" 进 行 连接 ， 实 现 不 间断 地 循环 输出 内 容 
document.write(<div id="templayer" style="position:absolute;z-index:1;visibility:hidden"></div>"); 
preTop=0; currentTop=0; 
function init(){ 
templayer.innerHTML="™; // 设 置 templayer 的 初始 值 为 空 
while(templayer.offsetHeight<marqueesHeightX{ /判断 当 templayer 的 高 度 小 于 marquees 的 高 度 时 
templayer.innerHTML+=marquees.innerHTML; /将 templayer 的 值 赋 给 marquees 


, 
marquees.innerHTML=templayer.innerHTML+templayer.innerHTML; // 将 templayer 的 值 累 加 
setlnterval("scrollup()",50); /间隔 50 毫秒 执行 一 次 scrollup() 函 数 
} 
function scrollup(}{ // 实 现 滚动 输出 
if(stopscroll==true) return; /| 判断 如 果 stopscroll==true， 不 执行 循环 


preTop=marquees.scrollTop; 

marquees.scrollTop+=1; 

if(preTop==marquees.scrollTop)\{ 
marquees.scrollTop=templayer.offsetHeight-marqueesHeight; 
marquees.scrollTop+=1; 


} 


</script> 
在 div 标签 中 ， 主 要 是 输出 数据 库 中 存储 的 公告 信息 ， 并 且 对 输出 的 信息 进行 截取 和 替换 ， 规 范 
输出 的 内 容 。div 标签 中 的 程序 代码 如 下 。 
<div id=marquees > <!-> 创 建 一 个 div 标签 <!--> 
<table width="200" height="25" border="0" align="center” cellpadding="0" cellspacing="0"> 
<?php // 从 数据 库 中 读 取 公 告 数据 


$sql=mysql_query("select id,title,createtime from tb_tell order by createtime desc limit 0,10",$conn); 
$info=mysql_fetch_array($sql); 





if($info==false}{ // 判 断 当 $info==false 时 执行 下 面 的 内 容 
?> 
<tr> 
<td height="25"><div align="center"><a href="#" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div></td> 
</tr> 
<?php  }else{ 
S$i=1; // 定 义 变量 $i=1 
dof /| 执行 do…while 循环 语句 
?> 
<tr> 


<td height="25" style="padding:4"> 
<a href="tellinfo.php?id=<?php echo $info[id];?>" class="a1"> 
<?php 
if($i==1){ / 削 断 当 $i==1 时 ,将 输出 的 内 容 设置 为 红色 
echo "<font color=red>"; 
echo $i.".&nbsp;"; 
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echo unhtml(msubstr($infoftitie],0,50)); /应 用 自 定义 函数 对 输出 的 内 容 进行 控制 

ifstrlen($infoftitle])>50){ // 当 输出 内 容 的 长 度 超过 50 个 字符 时 用 "…" 代 替 
en 

} 


echo "(".str_replace("-","/",$info[createtime]).")"; // 将 输出 的 公告 时 间 中 的 "-" 用 "" 替 代 
if($i==1X{ echo "</font>"; } 

?> </a> 
</td> 

</tr> 

<?php 
Bitt; 
}while($info=mysql_fetch_array($sql)); /ldo…while 循环 语句 结束 
} 


?> 
</table> 
</div> 


在 首页 中 使 用 滚动 条 是 一 个 比较 不 错 的 方法 ， 通 过 其 可 以 增加 网 页 的 动态 效果 ， 增 加 网 页 的 观赏 
性 ， 而 且 不 会 影响 到 网 页 的 浏览 速度 。 


22.7.3 ”前台 首页 的 实现 过 程 


开发 网 站 首页 主要 就 是 连接 数据 库 ， 从 数据 库 中 读 取 数 据 ， 最 后 应 用 循环 语句 将 数据 库 中 数据 输 
出 到 前 台 页 面 。 由 于 使 用 的 代码 较 多 ， 而 且 多 数 都 是 重复 使 用 ， 所 以 这 里 只 给 出 首页 中 公告 发 布 模块 
的 代码 。 

公告 发 布 模块 主要 实现 从 数据 库 中 读 取 公 告 数据 ， 将 数据 在 首页 中 滚动 输出 ， 并 且 对 公告 信息 的 
长 度 进行 控制 ， 保 证 内 容 的 整齐 、 规 范 。 具 体 详细 代码 可 以 参考 本 书 光盘 中 TM\02\bcty365\index.php 
文件 ，index.php 文件 的 部 分 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\22\bcty365\index.php 





<?php include_once("top.php"); // 获 取 头 部 文件 ?> 

“…// 省 略 部 分 代码 

<?php 
// 读 取 数据 库 中 公告 表 中 的 数据 
$sql=mysql_query("select id,title,createtime from tb_tell order by createtime desc limit 0,10",$conn); 
$info=mysql_fetch_array($sql); /| 执行 读 取 数据 表 中 数据 的 语句 
if($info==falseX{ // 如 果 返 回 值 为 空 则 执行 下 面 的 语句 

?> 

<tr> 


<td height="25"><div align="center"><a href="#" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div></td> 
</tr> 


<?php 
Jelse{ // 如 果 返 回 值 不 为 空 则 执行 下 面 的 do…while 循环 语句 
Si=1; 
do{ 
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2 
<tr> 
<td height="25" style="padding:4"><a href="tellinfo.php?id=<?php echo Sinfo[id];?>" class="a1"> 
<?php 
if($i==1X{ /判断 当 变 量 $i=1 时 ， 输 出 的 内 容 以 红色 字体 显示 
echo "<font color=red>"; 
F 
echo $i.".&nbsp;"; 
echo unhtml(msubstr($infoltitle],0,50)); 
ifstrlen($info[title])>50){ // 如 果 标 题 长 度 大 于 50 个 字符 ， 则 以 省 略 号 代替 
eho 
} 
echo "(".str_replace("-","/",$info[createtime]).")"， ”// 输 出 公告 发 布 的 时 间 ， 并 且 将 其 中 的 "/" 使 用 "-" 替 换 
if($i==1X{ 
echo "</font>"; 
} 
I 
</a> 
</td> 
</tr> 
<?php 
S$it+; // 变 量 自 加 1 
}while($info=mysql_fetch_array($sql)); /do…while 循环 语句 结束 
二 
人 /省 略 了 部 分 代码 
<?php 
include_once("bottom.php"); ll 调用 网 站 的 尾 文件 


?> 


22.8 注册 模块 设计 


22.8.1 注册 模块 概述 


BCTY365 网 上 社区 系统 为 了 更 好 地 与 广大 网 民 朋 友 进 行 交 流 和 沟通 ， 创 建 了 一 个 会 员 注册 模块 。 


通过 会 员 注册 模 
实现 与 用 户 的 长 
权限 。 在 本 系统 
中 发 表 留 言 ; 在 
块 的 运行 结果 如 
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中 注册 会 员 可 以 拥有 如 下 权限 : 在 本 社 








图 22.26 所 示 。 


块 ， 可 以 有 效 地 对 用 户 信息 进行 采集 ， 并 将 合法 的 用 户 信息 保存 到 指定 的 数据 表 中 ， 
期 沟通 和 交流 。 既 然 设 置 了 会 员 注册 模块 ， 那 么 在 系统 中 就 要 为 会 员 提供 一 些 特殊 的 





区 的 论坛 中 发 布 和 回复 帖子 ， 在 技术 支持 模块 


升级 下 载 模块 中 下 载 软件 升级 包 等 ; 而 且 可 以 进行 修改 密码 和 找 回 密码 。 用 户 注册 模 
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您 已 经 成 功 地 申请 了 本 站 的 用 己 名 ; PIG 程序 员 如 果 注册 为 本 站 的 用 户 ， 您 还 应 该 填写 以 下 信息 ? 
注意 : 以 下 注册 信 息 均 为 必 浅 内 容 ? 


真实 姓名 : 

性 别 : 攻 本 

邮箱 地 址 : Basiss em， 的 了 便于 工作 人 员 与 您 联系 ， 请 填写 正确 的 FE-ssil 地 址 ? ) 
[Er 
[于 组 成 ， 并 且 为 6 位 1) 
[只 能 由 梁子 组 成!) 


[长 看 市 


密码 保护 问题 : 我 晤 圳 欢 的 休 亲 运动 是 什么 ? 加 
您 的 答案 : (为 了 能 由 找 回 捷 失 的 密码 ， 请 记 住 读 答 案 ? ) 
效 给 码 : 





图 22.26 ”用户 注册 模块 的 运行 结果 


22.8.2 ”注册 模块 技术 分 析 


:会 员 注册 模块 中 ， 必 不 可 少 的 就 是 要 对 用 户 和 输 入 的 信息 进行 判断 ， 首 先 判断 用 户 填 写 的 注册 信 

息 中 哪些 是 必须 填写 的 ， 哪 些 可 以 不 填写 ， 然 后 进一步 判断 输入 的 信息 是 否 合理 合法 ， 例 如 ， 判 断 输 
入 的 邮编 的 格式 是 否 正确 ， 判 断 输 入 邮箱 的 格式 是 否 正确 等 。 对 表单 中 提交 数据 进行 判断 最 常用 的 办 
法 就 是 使 用 JavaScript 脚本 ， 也 可 以 使 用 正则 表达 式 。 下 面 讲解 在 本 模块 中 是 如 何 通过 JavaScript 实现 
表单 提交 数据 验证 。 

操作 原理 是 : 在 form 表单 中 调用 onsubmit 事件 ， 通 过 该 事件 调用 指定 的 JavaScript 脚本 ， 执 行 
chkinputO 自 定义 函数 ， 实 现 对 表单 中 提交 数据 的 验证 。 在 JavaScript 脚本 中 ， 实 现 对 表单 中 提交 数据 
进行 判断 ， 判 断 输入 的 内 容 是 否 为 空 ， 判 断 内 容 的 格式 是 否 正确 ， 如 果 正 确 则 继续 执行 ， 否 则 将 弹出 
提示 对 话 框 ， 并 将 鼠标 的 焦点 指定 到 出 错 的 位 置 。 具 体 的 JavaScript 脚本 代码 如 下 。 

<script language="JavaScript" type="text/javascript"> 





function chkinput(form){ /定义 一 个 函数 
if(form.tel.value=="™")}{ /判断 tel 文本 框 中 的 值 是 否 为 空 
alert(" 请 填写 联系 电话 "); // 如 果 为 空 则 输出 "请 填写 联系 电话 了 
form.tel.select(); /返回 到 tel 文本 框 
return(false); 
Ee /判断 email 文本 框 的 值 是 否 为 空 
alert(" 请 输入 E-mail 地 址 ""); // 如 果 为 空 则 输出 "请 输入 E-mail 地 址 " 
form .emailselect(): /返回 到 email 文本 框 
return(false); 
} 
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var i=form.email.value.indexOf("@"); 
var j=form.email.value.indexOf("."); 
/进一步 判断 邮箱 的 格式 是 否 正确 ,是 否 包含 "@" 和 "." 
读 (Gi<O)Ij>O)IG<O)X 
alert(" 请 输入 正确 的 E-mail 地 址 ""); 


form.email.select(); // 返 回 到 email 文本 框 
return(false); 
} 
return(true); /提交 表单 


b 

</script> 

上 述 代 码 中 ， 只 是 列举 了 JavaScript 中 的 部 分 内 容 ， 并 且 在 对 电话 号 码 进 行 判断 时 ， 只 是 判断 其 是 
否 为 空 ， 没 有 进一步 判断 电话 号 码 的 格式 是 否 正 确 。 如 果 想 要 更 加 准确 地 判断 电话 号 码 的 格式 是 否 正 
确 ， 可 以 采用 下 面 的 方法 : 通过 正则 表达 式 的 preg_match0 函 数 ， 在 表单 提交 处 理 页 中 对 电话 号 码 进行 
判断 。 

preg_match0 函 数 的 语法 格式 如 下 。 

int preg_match ( string pattern, string subject [, array matches [, int flags]] ) 

preg_match0 函 数 的 参数 说 明 如 表 22.1 所 示 。 


表 22.1 preg_match() 函 数 的 参数 说 明 





参数 说 明 

attern 必要 参数 。 需 要 匹配 的 正则 表达 式 

subject 必要 参数 。 输 入 的 字符 串 

人 可 选 参数 。 输 出 的 搜索 结果 的 数组 ， 如 $out[0] 将 包含 与 整个 模式 匹配 的 结果 ，$out[]1] 将 包含 与 
第 一 个 捕获 的 括号 中 的 子 模式 所 匹配 的 结果 ， 以 此 类 推 

全 可 选 参数 。 标 记 : PREG_ OFFSET CAPTURE， 对 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字 


符 串 偏 移 量 ， 本 标记 自 PHP 4.3.0 起 可 用 





通过 preg_matchO 函 数 判断 电话 号 码 的 格式 是 否 正确 的 方法 如 下 : 首先 定义 一 个 用 于 判断 电话 号 码 
格式 的 正则 表达 式 。 代 码 如 下 。 

人 (df3})Cd(8D)SIA(d(4}-)Gdf{7DSIACd(4F-)Odf8DS/ 

正则 表达 式 的 功能 分 析 如 下 : 使 用 “^” 和 “$” 对 字符 串 进行 边界 的 限制 ， 对 区 号 从 字符 串 的 开 
始 进行 匹配 ， 对 其 他 号 码 从 字符 串 的 末尾 开始 进行 匹配 ; 将 括号 “0” 中 的 内 容 作 为 一 个 原子 使 用 ; 使 
用 “\d” 来 匹配 一 个 数字 ， 区 号 为 3 或 4 个 数字 ， 其 他 数字 为 7 或 8 个 ; 使 用 “和 ”来 对 前 字符 进行 重 
复 匹 配 ; 使 用 “|” 对 匹配 的 模式 进行 选择 ， 分 成 3 个 模式 。 
然后 将 该 正则 表达 式 应 用 到 preg_match0 函 数 中 ,对 表单 提交 的 电话 号 码 进行 判断 ， 如 果 正 确 则 继 
续 执行 ， 否 则 弹出 提示 信息 ， 并 返回 到 表单 提交 页 ， 代 码 如 下 。 














<?php 
S$tel="0431-84978981"; /定义 一 个 电话 号 码 的 变量 
if(preg_match(", rt he 1})$/",S$tel,S$counts)\{ 
echo "您 输入 的 电话 号 码 格式 正确 // 输 出 字符 串 
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Jelse{ 
echo "<script>alert(' 您 输入 的 电话 号 码 的 格式 不 正确 !);history.back()</script>"; 
} 


?> 


22.8.3 ”注册 模块 的 实现 过 程 


注册 模块 的 实现 过 程 非常 简单 ， 首 先 阅读 注册 服务 条 款 ， 然 后 填写 用 户 注册 的 用 户 名 和 密码 ， 提 
交 后 由 系统 判断 输入 的 用 户 名 是 否 被 占用 ， 如 果 未 被 占用 则 可 以 继续 注册 ， 填 写 详细 的 注册 信息 ， 将 
数据 提交 到 表单 处 理 页 进行 处 理 ， 最 后 将 用 户 注册 的 信息 保存 到 指定 的 数据 表 中 。 用 户 注册 模块 的 实 
现 过 程 主要 由 3 个 文件 完成 :register.php 用 于 输出 注册 服务 条 款 ， 以 及 填写 注册 的 用 户 名 和 密码 ， 并 
且 判 断 注册 的 用 户 名 和 密码 是 否 被 占用 ; getuserinfo.php 文件 用 于 填写 详细 的 注册 信息 ， 并 且 在 表单 中 
应 用 数字 验证 码 技术 ; savereginfo.php 文件 用 于 对 表单 中 提交 的 数据 进行 处 理 , 将 数据 保存 到 指定 的 数 
据 表 中 。 

在 savereginfo.php 文件 中 ， 首 先 连接 数据 库 ， 然 后 获取 到 表单 中 提交 的 数据 ， 并 且 判 断 提交 的 用 
户 名 是 否 被 占用 ， 最 后 将 提交 的 数据 进行 处 理 ， 并 将 数据 保存 到 指定 的 数据 表 中 。 程 序 代码 如 下 。 

代码 位 置 ， 光 盘 \TM\sl\22\bcty365\saverreginfo.php 








<?php session_start(); /初始 化 session 变量 

include_once("conn/conn.php"); /| 连接 数据 库 

$usernc=trim($_POST[usernc]); // 获 取 注 册 的 用 户 名 

// 判 断 指定 的 用 户 名 是 否 存在 

$sql=mysql_query("select usernc from tb_user where usernc=". $usernc."",$conn); 

$info=mysql_fetch_array($sql); // 按 指定 条 件 检索 数据 信息 

if($info!=falseX{ // 如 果 查 询 结果 不 为 空 ， 则 执行 以 下 操作 
echo "<script language='javascript'>alert(' 对 不 起 ， 该 昵称 已 被 其 他 用 户 使 用 !");history.back();</script>"; 
exit; 

} 

$xym=trim($_POST[xym]); /去 除 变量 两 边 的 空格 

$num=$_POSTInum]; /接收 变量 值 


ifstrval($xym)l=strval($num)){ 
echo "<script>alert(' 验 证 码 输入 错误 !");window.location.href='register.php';</script>"; 


exit;} 

// 对 表单 提交 的 数据 进行 处 理 

$truepwd=trim($_POST[pwd1]); // 获 取 真 实 密码 
S$pwd=md5($truepwd); /| 获取 加 密 密 码 
S$truename=trim($_POST[truename]); // 获 取 真 实 姓名 
S$email=trim($_POST[email); /获取 邮箱 地 址 
$sex=$_POST[sex]; // 获 取 性 别 
Stel=trim($_POST[tel); /获取 电话 
$yb=trim($_POST[yb]); // 获 取 邮 政 编码 
$qq=trim($_POST[qq]); /获取 QQ 
$address=trim($_POST[address]); /获取 地 址 
$question=trim($_POST[question]); /获取 提示 问题 
S$answer=trim($_POST[answer]); /获取 问题 答案 
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S$ip=getenv("REMOTE_ADDR"); // 获 取 客 户 端的 IP 
Slogintimes=1; /指定 访问 次 数 
S$regtime=date("Y-m-j H:i:s"); 1/ 获取 时 间 
S$lastlogintime=$regtime; 

$usertype=0; /指定 用 户 类 型 ， 黑 认为 0 
$photo=$_POST["photo"]; // 获 取 头 像 

// 将 表单 中 提交 的 数据 存储 到 数据 表 中 


if(mysql_query("insert into 
tb_user(usernc,truename,pwd,email,sex,tel,qq,address,logintimes,regtime,lastlogintime.,ip,usertype,yb,question 
,answer,truepwd,photo) 
values('$usernc','$truename','$pwd','$email','$sex','$tel','$qq','$address','$logintimes','$regtime','$lastlogintime’,' 
Sip'",'$usertype','$yb','$question','$answer','$truepwd','$photo')", $conn)){ 

session_register("unc"); 

$_SESSION["unc"]=$usernc; 

echo "<script>alert(' 注 册 成 功 !");window.location.href='index.php';</script>"; 


Jelse{ // 如 果 添 加 操作 失败 ， 则 给 出 提示 
echo "<script language='javascript'>alert(' 对 不 起 ,注册 失败 "");history.back();</script>"; 
exit; /退出 

} 

?> 


22.9 技术 支持 模块 设计 


技术 支持 模块 主要 是 从 浏览 者 的 角度 进行 设计 ， 存 储 大 量 技术 问题 的 解决 方案 数据 ， 为 浏览 者 查 
阅 提供 方便 ， 而 且 设计 一 个 企业 与 客户 沟通 的 平台 ， 能 够 随时 了 解 客 户 或 者 会 员 的 意见 和 需求 。 


22.9.1 技术 支持 模块 概述 


技术 支持 模块 主要 由 3 个 子 模块 组 成 ， 包 括 常 见 问题 、 客 户 反馈 和 联系 方式 。 常 见 问题 模块 主要 
用 于 展示 编程 中 一 些 常见 问题 的 解决 方案 或 者 方法 ， 为 浏览 者 解决 编程 中 的 疑难 问题 提供 方便 ; 客户 
反馈 模块 主要 用 于 收集 和 获取 来 自 客户 的 需求 和 意见 ， 联 系 方式 模块 主要 用 于 展示 企业 的 形象 和 具体 
的 联系 方式 。 





22.9.2 ”技术 支持 模块 技术 分 析 


技术 支持 模块 中 在 对 常见 问题 解决 方案 的 数据 进行 输出 时 ， 使 用 了 分 页 处 理 技术 ， 该 技术 的 设计 
思路 是 : 从 数据 库 中 读 取 数据 ， 获 取 数据 总 量 ， 在 每 页 中 显示 20 条 数据 ， 根 据 数 据 总 量 和 每 页 显示 的 
条 数 对 数据 进行 分 页 处 理 ， 计 算出 有 多 少 页 和 当前 显示 的 页 码 ， 实 现 首页 、 上 一 页 、 下 一 页 和 尾 页 之 
间 的 页 面 跳 转 。 具体 的 设计 思路 可 以 参考 cjwtphp 文件 中 的 代码 注释 和 代码 贴 士 。 cjwtphp 文件 的 程序 
代码 如 下 。 
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代码 位 置 : 光盘 \TM\sl22\bcty365\cjwt.php 


<?php 
$sql=mysql_query("select count(*) as total from tb_cjwt",$conn); 。“”// 读 取 数 据 库 中 数据 
$info=mysql_fetch_array($sql); // 返 回 数据 


S$total=$info[total]; 
// 判 断 字段 total 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 


if($total==0X{ 
1 
<tr> 
<td height="22" colspan="2"><div align="center"> 对 不 起 ， 暂 无 常见 问题 !</div></td> 
</tr> 
<?php 
}else{ // 如 果 不 为 空 ， 则 执行 下 面 的 内 容 
if(lisset($_GET['page"]) || !is_numeric($_GETT'page"])X( / 尖 | 断 $_GET 获取 的 page 的 值 是 否 存在 
S$page=1; // 如 果 不 存在 ， 则 设置 变量 的 值 为 1 
}else{ 
Spage=intval($_GET["page"]); // 如 果 存 在 ， 则 获取 变量 $_GET 的 值 
// 设 置 变 量 $pagesize, 每 页 显示 的 数据 数据 量 为 20 
$pagesize=20; 
if($total%$pagesize==0X{ /如 果 变 量 的 值 为 0 
$pagecount=intval($total/$pagesize); // 获 取 变 量 的 整数 值 
}else{ 
S$pagecount=ceil($total/$pagesize); // 如 果 不 为 0 则 获取 实际 的 整数 值 


} 
// 读 取 数 据 库 中 数据 ， 按 照 时 间 进 行 降 需 排列 
$sql=mysql_query("select * from tb_cjwt order by createtime desc limit ".($page-1)*$pagesize.", 
$pagesize ",$conn); 
While($info=mysql_fetch_array($sql))f 
?> 
加 // 省 略 部 分 代码 
<table width="600" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="479"><div align="left">&nbsp;&nbsp; 共 有 常见 问题 &nbsp;<?php echo S$total;?>&nbsp; 条 &nbsp; 
每 页 显示 &nbsp;<?php echo $pagesize;?>&nbsp; 条 &nbsp; 第 &nbsp;<?php echo $page;?>&nbsp; 页 / 共 
&nbsp;<?php echo $pagecount;?>&nbsp; 页 </div></td> 
<td width="269"><div align="right"> 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=1" class="a1"> 首 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 


if($page>1) / 淹 断 如 果 页 码 大 于 1 
echo $page-1; /输出 前 一 页 

else 
echo 1; 


?>" class="a1"> 上 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 


if($page<$pagecount) // 如 果 页 码 小 于 总 页 数 
echo $page+1; 

else 
echo $pagecount; // 输 出 下 一 页 


?>" class="a1"> 下 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php echo S$pagecount;?>" 
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class="a1"> 尾 页 </a>&nbsp;&nbsp;</div> 
</td> 
</tr> 
</table> 


22.9.3 ”常见 问题 的 实现 过 程 


常见 问题 子 模块 实现 的 主要 功能 是 展示 出 数据 库 中 存储 的 有 关 编 程 中 过 到 的 常见 问题 及 解决 方 
案 。 其 运行 结果 如 图 22.27 所 示 。 


人 技术 支持 >> 


问 题 : ”如何 解 决 输出 数据 库 中 数据 乱码 的 问题 ? 


在 连接 数据 库 的 文件 中 , 指定 数据 库 中 数据 的 编码 格式 为 “gb2312” 
这 里 活 接 一 个 用 户 各 为 root, 密码 为 zoo 多 数据库 服 务 器 ,指定 连接 有 数据库 为 各 _xoot, 并 且 指定 数据 库 
中 编码 格式 为 2312 








通过 该 方法 ,就 可 以 各 免 在 输出 数据 库 中 的 数据 时 出 现 乱 碍 
22.27 ”常见 问题 模块 的 运行 结果 


该 模块 由 两 个 文件 组 成 ， 一 个 是 jwtphp 文件 ， 用 于 存储 创建 问题 数据 ， 详 细 内 容 可 以 参考 22.9.2 
节 ;， 另 一 个 是 lookcjwtphp 文件 ， 用 于 输出 cjwt.php 文件 中 对 应 问题 的 详细 介绍 和 解决 方案 。 其 代 
码 如 下 。 


代码 位 置 : 光盘 \TMNsl22\bcty365ookcjwtphp 


<?php 
include_once("conn/conn.php"); /| 与 数据 库 建 立 连 接 
include_once("function.php"); /调用 自 定义 函数 


// 读 取 tb_cjwt 表 中 的 数据 ,条 件 为 id="$_GET[id]" 
$sql=mysql_query("select * from tb_cjwt where id=".$_GET["id]."",$conn); 
$info=mysql_fetch_array($sql); 


?> 
<table width="635" height="100" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="94" height="30"><div align="center"><strong> 问 &nbsp;&nbsp; 题 : </strong></div></td> 
<td width="541"><?php echo unhtml($info["question"]); // 输 出 问题 的 详细 内 容 ?></td> 
<htr> 
<tr> 
<td height="70"><div align="center"><strong> 解 &nbsp;&nbsp; 答 : </strong></div></td> 
<td height="70">&nbsp;<?php echo unhtml($info["answer]); /| 输出 问题 的 解决 方案 ?></td> 
</tr> 
</table> 
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22.9.4 客户 反馈 的 实现 过 程 


客户 反馈 子 模块 为 客户 提供 一 个 反馈 意见 和 提出 要 求 的 平台 , 并 且 将 提交 的 信息 存储 到 数据 库 中 。 
其 运行 结果 如 图 22.28 所 示 。 


(四 技术 支持 >》 和 常见 问题 客户 反馈 联系 方式 


主 题 : 隐 得 河 册 什么 时 候 可 以 上 市 
类 别 : 我 的 留言 局 





| 请问 你 们 的 编程 词典 什么 时 候 可 以 上 市 ? 司 
内 容 
剧 
效 验 码 : EE 55 重 写 


图 22.28 客户 反馈 模块 的 运行 结果 
该 功能 只 对 本 网 站 中 的 会 员 开 通 ， 即 只 有 以 会 员 身 份 登录 的 用 户 才 具 有 反馈 信息 的 权限 ， 其 中 在 
对 提交 表单 的 细节 处 理 上 使 用 JavaScript 脚本 来 验证 表单 中 的 值 是 否 为 空 ， 而 且 还 使 用 数字 验证 码 技 
术 。 对 表单 中 提交 的 数据 进行 处 理 是 在 saveleaveword.php 文件 中 完成 ， 该 文件 主要 用 于 对 表单 中 提交 
的 数据 进行 处 理 ， 将 数据 存储 到 数据 库 中 ， 其 代码 如 下 。 
代码 位 置 : 光盘 \TM\sl\22\bcty365\saveleaveword.php 


<?php session_start(); /初始 化 一 个 session 变量 
$xym=$_POST[xym]; // 获 取 $_POST 提交 的 值 
if($xym!=$_SESSION["autonum"]X{ /判断 验证 码 是 否 正确 
echo "<script>alert(' 效 验 码 输入 错误 !");history.back();</script>"; 
exit; 
要 
Stitle=$_POSTT'title"]; /获取 反馈 信息 的 标题 
S$content=$_POST["content"]; // 获 取 反 馈 信息 的 内 容 
S$type=$_POST["type"]: /获取 反馈 信息 的 类 型 
include_once("conn/conn.php"); /与 数据 库 建立 连接 
$sql=mysql_query("select id from tb_user where usernc=".$_SESSION["unc"]."",$conn); // 读 取 数 据 库 中 数据 
$info=mysql_fetch_array($sql); // 获 取 结 果 集中 的 数组 
$userid=$info["id"]; 
/向 数据 库 中 添加 数据 


if(mysql_query("insert into tb_leaveword(userid,type,title,content,createtime) 





values('$userid','$type,",'S$title','$content',".date("Y-m-j H:i:s").")",$conn)){ 

echo "<script>alert(' 留 言 发 表 成 功 !");history.back();</script>"; /添加 操作 成 功 ， 给 出 提示 信息 
Jelse{ 

echo "<script>alert(' 留 言 发 表 失败 !");history.back();</script>"; // 添 加 操作 失败 ， 给 出 提示 信息 
es 
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22.10 在线 订购 模块 设计 


在 线 订购 模块 的 功能 是 实现 在 线 购买 企业 推出 的 软件 产品 ， 其 操作 的 流程 主要 通过 购物 车 和 订单 
管理 来 实现 。 


22.10.1 在 线 订 购 模块 概述 


在 线 订购 的 功能 对 所 有 访问 网 站 的 人 开放 ， 没 有 任何 的 权限 限制 。 其 操作 流程 如 图 22.29 所 示 。 


浏览 者 





22.29 在 线 订购 模块 的 操作 流程 
22.10.2 在线 订购 模块 技术 分 析 


在 线 订购 管理 模块 中 ， 不 可 缺少 的 一 项 内 容 就 是 对 订单 进行 打印 。 下 面 就 来 讲解 一 下 订单 打印 功 
能 的 实现 方法 。 在 线 订购 管理 模块 中 运用 的 是 WebBrowser 打印 方法 。WebBrowser 是 正 内 置 的 浏览 器 
控件 ， 无 须 用 户 下 载 。 其 优点 是 客户 端 独立 完成 打印 目标 文档 的 生成 ， 减 轻 服务 器 负荷 :缺点 是 源 文 
档 的 分 析 操 作 复 杂 ， 并 且 要 对 源 文 档 中 要 打印 的 内 容 进行 约束 。 

下 面 介绍 WebBrowser 控件 的 具体 参数 ， 如 表 22.2 所 示 。 


表 22.2 WebBrowser 控件 的 具体 参数 说 明 











参数 名 称 说 明 
document.all. WebBrowser.Execwb(7.1): 表示 打印 预览 
document.all. WebBrowser.Execwb(6.1): 表示 打印 
document.all. WebBrowser.Execwb(6.6): 表示 直接 打印 
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续 表 
参数 名 称 说 有明 
document.all. WebBrowser Execwb(8.1): 表示 页 面 设 置 
document.all. WebBrowserExecwb(1.1): 打开 页 面 
document.all. WebBrowser.Execwb(2.1): 关闭 所 有 打开 的 正 窗口 
document.all. WebBrowser.Execwb(4.1): 保存 网 页 
document.all. WebBrowser.Execwb(10.1): 查看 页 面 属性 
document.all.WebBrowser.Execwb(17.1): 全 选 
document.all. WebBrowser.Execwb(22.1): 刷新 
document.all. WebBrowser. Execwb(45,1): 关闭 窗 体 无 提示 


该 技术 的 实现 原理 是 :首先 通过 onClick 事件 调用 一 个 JavaScript 脚本 ,然后 执行 openprintwindowO 
函数 ， 将 指定 的 变量 值 传递 到 订单 打印 页 中 (printwindow.php) ， 最 后 在 订单 打印 页 中 实现 打印 及 打 
印 预览 的 功能 。 调 用 JavaScript 脚本 和 执行 openprintwindow0 函 数 在 shopping_dd.php 页 中 完成 。 其 关 
键 代 码 如 下 。 

代码 位 置 ， 光 盘 \TM\sl\22\bcty365\shopping_dd.php 

<script language="javascript"> 

function openprintwindow(x,y,zX{ /定义 一 个 函数 ， 获 取 传 递 的 参数 
/通过 window 对 象 中 的 open 方法 打开 一 个 新 窗口 ， 并 设置 其 属性 
window.open("printwindow.php?ddno="+x+"&pv="+z,"newframe","top=200,left=200,width=635， 
height="+(230+20*y)+",menubar=no,location=no,toolbar=no,scrollbars=no,status=no"); 
上 
</script> 

< 通过 onclick 事件 调用 JavaScript 脚本 ， 传 递 参数 值 。 当 参数 z 的 值 为 "p" 时 执行 打印 功能 

<td width="75"> <img src="images/bg_14(11).jpg" width="69" height="20" 

onclick="javascript:openprintwindow('<?php echo base64_decode($_GET["ddno"]);?>''<?php echo 

S$gnum;?>','p')" style="cursor:hand"/></td> 

<!-- -一 一 通过 onclick 事件 调用 JavaScript 脚本 ， 传 递 参数 值 。 当 参数 z 的 值 为 "v" 时 执行 打印 预览 功能 

<td width="90"><img src="images/bg_14(12).jpg" width="69" height="20" 

onclick="javascript:openprintwindow('<?php echo base64_decode($_GET["ddno"]);?>','<?php echo 

S$gnum;?>",'Vv')" 

style="cursor:hand"/></td> 


订单 的 打印 和 打印 预览 的 功能 在 printwindow.php 页 中 完成 ， 首 先 编写 一 个 实现 打印 预览 功能 的 
JavaScript 脚本 ， 然 后 建立 HIML 的 object 标签 ， 调 用 WebBrowser 控件 ， 最 后 获取 变量 传递 的 值 ， 当 
变量 的 值 为 时 执行 打印 功能 ， 当 变量 的 值 为 时 执行 打印 预览 的 功能 。printwindow.php 页 的 关键 代 
码 如 下 。 

代码 位 置 : 光盘 \TM\sl\22\bcty365\printwindow.php 


-> 











-> 


<script> 
function printview()f /定义 一 个 函数 
document.all.WebBrowser1.ExecWB(7,1) ; /| 执行 WebBrowser 控件 ， 实 现 打印 预览 
window.close(); /关闭 窗口 
上 
</script> 
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< -一 一 一 一 一 一 一 一 一 建立 HTML 的 object 标签 ， 调 用 WebBrowser 控件 -一 -一 ---------- 一 > 
<object ID="WebBrowser1' WIDTH=0 HEIGHT=0 
CLASSID='CLSID:8856F961-340A-11D0-A96B-00C04FD705A2'></object> 


< 上 一 一 在 body 中 调用 onload 事件 ， 执 行 打 印 操 作 一 一 一 一 一 一 一 一 一 一 一 > 
<body topmargin="0" leftmargin="0" bottommargin="0" onLoad=" 
<?php 
if($_GET["pv"]=="p"X ”// 判 断 当 变 量 值 为 "p" 时 ， 执 行 打印 操作 
> 
window.print(); 
<?php 
Jelseif($_GET["pv"]=="v"X// 判 断 当 变 量 值 为 "vV" 时 ， 执 行 打印 预览 操作 
2 
printview() 
<?php} ?> 
ke 


订单 打印 操作 的 运行 结果 如 图 22.30 所 示 。 

















图 22.30 订单 打印 操作 的 运行 结果 
22.10.3 ”购物 车 的 实现 过 程 
购物 车 的 功能 是 临时 储存 用 户 选 购 的 商品 ， 用 户 可 以 对 购物 车 中 的 商品 进行 添加 、 修 改 、 删 除 和 


更 新 操作 ， 也 可 以 选择 进行 结算 。 其 运行 的 结果 如 图 22.31 所 示 。 











责骂 各 单价 (元 7 要 量 ( 个 ) | 给 作 
3 区 二 类 量 | 扫除 该 项 
本 333.00 厂 | 页 下 生 | 到 法 项 
继续 购买 ” 洁 宝 更 物 车 商品 全 额 总 计 666.00 元 EE 











22.31 购物 车 的 运行 结果 
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购物 车 功能 实现 的 第 一 步 是 为 想 要 购买 产品 的 用 户 分 配 一 辆 购物 车 ， 使 其 能 够 记录 自己 已 经 选 购 
的 产品 。 其 工作 的 原理 与 超市 中 顾客 使 用 购物 车 进行 购物 是 相同 的 ， 只 是 这 里 使 用 的 不 是 真正 意义 上 











的 购物 车 ， 而 是 两 个 session 变量 ， 一 个 存储 





lj 户 选 购 商 品 的 ID〈$goodsid) ， 另 一 个 存储 用 户 选 购 该 


商品 的 数量 ($goodsnum) 。 如 果 用 户 在 一 次 购物 中 选 购 多 种 不 同类 的 商品 ， 则 使 用 “@” 对 不 同类 商 
品 的 不 同 ID 和 数量 进行 分 隔 。 例 如 ， 用 户 选 购 的 不 同类 商品 id 为 1、2、3， 则 session 变量 $goodsid 
中 存储 的 值 为 “1@2@3@”; 其 中 同一 种 商品 不 能 购买 两 次 ， 如 果 想 要 购买 多 个 同 种 产品 ， 可 以 在 购 








物 车 中 更 改 购买 商品 的 数量 。 在 本 案例 中 ， 购 
shopping_cart_first.php 文件 的 代码 如 下 。 


物 车 的 分 配 功 能 通过 shopping_cart_first.php 文件 来 完成 ， 


代码 位 置 ， 光盘 \TM\sl\22\bcty365\shopping_cart_first.php 


<?php session_start(); 
session_register("goodsid"); 
session_register("goodsnum"); 


// 初 始 化 session 变量 
/创建 一 个 session 变量 
// 创 建 一 个 session 变量 


if($_SESSION["goodsid"]==" && $_SESSION["goodsnum"]=="X{ ”// 判 断 session 变量 中 的 值 是 否 为 空 


$_SESSION["goodsid"]=$_GET["id"]."@"; // 如 果 为 空 则 将 商品 的 ID 赋 给 变量 
$_SESSION["goodsnum"]="1@"; // 将 商品 数量 设置 为 1@ 
}else{ 
S$array=explode("@",$_SESSION["goodsid"]); // 如 果 不 为 空 ， 则 使 用 @ 分 隔 不 同 的 商品 ID 


// 判 断 如 果 获 取 的 ID 在 session 变量 中 已 经 存在 ， 则 提示 该 商品 已 经 被 放 入 购物 车 


if(in_array($_GET["id"], $array)}{ 


echo "<script>alert(' 该 编程 词典 已 经 被 放 入 购物 车 ! ');history.back();</script>"; 


exit; 


} 


$_SESSION["goodsid"].=$_GET["id"]."@"; // 为 session 变量 赋值 


$_SESSION["goodsnum"].="1@"; 


// 将 商品 放 入 购物 车 中 ， 并 跳 转 到 购物 车 页 


/为 session 变量 赋值 


echo "<script>window.location.href='shopping_cart.php';</script>"; 


> 


在 实现 购物 车 的 分 配 和 添加 商品 的 功能 后 ， 接 下 来 要 做 的 就 是 查看 购物 车 中 的 商品 ， 即 实现 购物 
车 中 商品 展示 的 功能 。 在 购物 车 的 商品 展示 中 ， 可 以 实现 很 多 的 操作 ， 如 清空 购物 车 、 删 除 购买 商品 、 





更 改 购买 商品 数量 、 继 续 购 物 和 结算 。 


购物 车 商品 展示 的 功能 主要 通过 shopping_cart.php 文件 来 完成 ， 首 先 从 session 变量 中 读 取 商品 的 
ID 和 数量 ， 然 后 根据 商品 的 ID 循环 输出 购物 车 中 的 商品 ， 最 后 以 商品 DD 为 标识 符 设置 不 同 的 超级 链 
接 ， 执 行 删除 商品 或 者 更 改 购买 商品 数量 等 操作 。 其 代码 如 下 。 

代码 位 置 : 光盘 \TMN\sl\22\bcty365\shopping_cart.php 

<table width="720" height="47" border="0" align="center" cellpadding="0" cellspacing="1" bgcolor="#999999"> 


<?php 
$array=explode("@",$_SESSION["goodsid"]); // 读 取 session 变量 中 的 商品 ID， 以 @ 进 行 分 隔 
S$arraynum=explode("@",$_SESSION["goodsnum"]); // 读 取 session 变量 中 的 商品 数量 ， 以 @ 进 行 分 隔 
$markid=0; // 创 建 变 量 ， 初 始 值 为 0 
for($i=0;$i<count($array); $i++{ /应 用 for 循环 语句 循环 输出 商品 ID 的 值 
if($array[$i]!="™"){ // 判 断 如 果 商 品 ID 的 值 不 为 
Smarkid++; 1/ 增 加 变量 $markid 的 值 
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} 

} 

if($markid==0X{ /判断 如 果 变量 $markid 的 值 为 空 则 输出 下 面 的 内 
容 
?> 

<tr> 

<td height="22"” colspan="4”bgcolor="#FFFFFF"><div align="center"> 对 不 起 您 的 购物 车 中 暂 无 商品 信 
息 !</div></td> 


</tr> 
<?php 
}else{ // 如 果 $markid 的 值 不 为 空 ， 则 执行 下 面 的 内 容 
$totalprice=0; /创建 变量 $totalprice， 初 始 值 为 0 
for($i=0;$i<count($array);$i++)f // 循 环 输出 数组 中 的 商品 ID 值 
if$array[$il!=")f 
// 根 据 获取 的 商品 ID 的 值 ， 从 数据 库 中 获取 对 应 产品 的 信息 
$sqlcart=mysql_query("select * from tb_bccd where id=".$array[$i]."",$conn); 
$infocart=mysql_fetch_array($sqlcart) 
?> 
<tr> 


<form name="form<?php echo $array[$i]?>" method="post" action="changegoodsnum.php"> 

<td height="22" bgcolor="#FFFFFF">&nbsp;<?php echo unhtml($infocart["bccdname"]);?></td> 

<td height="22" bgcolor="#FFFFFF"><div align="center"><?php echo 
number_format($infocart["price"],2);?></div></td> 

<td height="22" bgcolor="#FFFFFF"><div align="center"><input type="text" name="goodsnum" 
value="<?php echo $arraynum["$i"];?>" class="inputcss" size="8" ><input type="hidden" name="id" 
value="<?php echo 
S$infocart["id"];?>" ></div></td> 

<td height="22" bgcolor="#FFFFFF"><div align="center"><a href="javascript:form<?php echo 
$array[$i]?>.submit();” class="a1"> 更 改 数 量 </a>&nbsp;|l&nbsp;<a href="delgoods.php?id=<?php echo 


Sinfocart["id"]:?>" 
class="a1"> 删 除 该 项 </a></div></td> 
</form> 
</tr> 
<?php 
$totalprice+=$infocart["price"]*$arraynum["$i"]; 
} 
} 
} 
?> 
</table> 


22.10.4 ”商品 订单 的 实现 过 程 


在 确定 所 要 购买 的 商品 之 后 ， 接 下 来 要 做 的 就 是 进行 购物 结算 ， 填 写 用 户 购物 订单 ， 将 订单 保 
存 到 数据 库 中 ， 并 且 随 机 生成 一 个 订单 号 ， 作 为 订单 的 唯一 标识 。 生 成 订单 的 运行 结果 如 图 22.32 
所 示 。 
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四 室 当 前 的 位 于 ?> 订 音 处理 


吉 李 过 明日 科技 表 限 公司 -本 和 元 订单 





ET 





么 过 要 支付 的 全 可 合计 为 : 


本 有 | HJ 轩 | 





图 22.32 ”生成 订单 的 运行 结果 
订单 提交 以 后 用 户 可 以 选择 汇款 的 方式 : 





-是 选择 网 上 支付 ， 那 么 将 跳 转 到 企业 指定 的 网 上 银行 


进行 汇款 ， 汇 款 的 操作 将 在 企业 指定 的 网 上 银行 中 进行 ， 具 体 实 现 过 程 将 在 22.10.5 节 进 行 介绍 ; 二 是 

选择 到 指定 的 银行 向 企业 提供 的 账号 中 汇款 。 企 业 将 在 收 到 汇款 后 按照 用 户 指定 的 地 址 和 方式 将 产品 

送 到 。 商 品 订单 的 生成 和 处 理由 shopping cart_getuserinfo php 和 savebuyuser.php 文件 来 完成 。 
订单 处 理由 savebuyuser.php 完成 ， 首 先 连接 数据 库 ， 随 机 生成 一 个 订单 号 ， 然 后 获取 购物 车 中 的 


商品 信息 ， 最 后 将 商品 信息 和 订单 号 存储 到 数据 库 中 。 其 代码 如 下 。 

代码 位 置 ， 光 盘 \TM\sl\22\bcty365\savebuyuser.php 
<?php session_start(); 
include_once("conn/conn.php"); 
$ddnumber=substr(date("YmdHis"),2,8).mt_rand(100000,999999); 
$sql=mysql_query("select * from tb_city where id=".$_POST["city"]."",$conn); 
$info=mysql_fetch_array($sql); 
说 $shfs=="1 

$yprice=$info[pt]; 

$shfs=" 普 通 邮递 "; 
}elseif($shfs=="2"){ 

$yprice=$info[kd]; 

$shfs=" 邮 政 特 快 专递 EMS"; 
} 
$array=explode("@",$_SESSION["goodsid"]); 
$arraynum=explode("@",$_SESSION["goodsnum"]); 
Stotalprice=0; 
for($i=0;$i<count($array):$i++}{ 

if($array[$i]!=""){ 


/初始 化 session 变量 

// 连 接 数 据 库 

// 随 机 生成 订单 号 

// 读 取 数 据 库 中 的 城市 信息 


// 判 断 用 户 选择 的 送 货 方式 


// 以 @ 来 分 隔 session 变量 中 存储 的 商品 ID 
/以 @ 来 分 隔 session 变量 中 存储 的 商品 数量 


/循环 读 取 数 组 中 商品 的 ID 


$sqlcart=mysql_query("select * from tb_bccd where id=".$array[$i]."",$conn); 


$infocart=mysql_fetch_array($sqlcart): 
$totalprice+=$infocart["price"]*$arraynum["$i]; 
} 
直 
$totalprice=$totalprice+$yprice; 
/将 表单 中 提交 的 数据 存储 到 数据 库 中 


/获取 汇款 金额 
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if(mysql_query("insert into 
tb_dd(ddnumber,recuser,sex,address,yb,qq,email,mtel,gtel,shfs,spc,slc,yprice,totalprice,createtime,cityid) 
values(".$ddnumber.",".$_POST["recuser"].",".$_POST["sex"].",".$_POST["address"].",".$_POST["yb"].",".$ 
_POST['qq"].",".$_POST["email"].",".$_POST['mtel"].",".$_POST["gtel"].",".$shfs.",".$_SESSION["goodsid"] 
.",".$_SESSION["goodsnum"].",".$yprice.”",”.$totalprice.",".date("Y-m-d 
H:i:s").",".$_POST["city"].")",$conn)X{ 

session_unregister("goodsid"); /注销 session 变量 goodsid 

session_unregister("goodsnum"); /注销 session 变量 goodsnum 

echo "<script>window.location.href='shopping_dd.php?ddno=".base64_encode($ddnumber).";</script>"; 
Jelse{ 

echo "<script>alert(' 订 单 信息 保存 失败 ， 请 重 试 !);</script>"; 


?> 


22.10.5 在线 支付 的 实现 


所 谓 在 线 支付 就 是 客户 端 (金融 机 构 需 客户 端 安装 由 金融 机 构 签发 的 数字 证 书 ， 信 用 卡 免 安装 ) 
将 支付 信息 加 密 后 通过 互联 网 传送 到 支付 网 关 (支付 网 关 是 解决 网 络 上 安全 支付 问题 的 交易 平台 ， 位 
于 互联 网 和 传统 的 金融 机 构 内 部 网 之 间 ， 其 主要 作用 是 将 互联 网 和 金融 网 络 安 全 地 连接 起 来 ， 将 不 安 
全 的 网 上 交易 信息 传 给 安全 的 金融 网 络 ， 起 到 隔离 和 保护 金融 网 络 的 作用 ) ， 同 时 金融 机 构 网 上 支付 
系统 反馈 有 关 支付 信息 ， 客 户 确认 无 误 后 进行 支付 确定 ， 支 付 网 关 负 责 商 户 网 上 交易 资金 的 清算 ， 并 
根据 商户 提供 的 开户 行 、 账 号 等 结账 信息 将 网 上 消费 款项 汇总 划 入 商户 账户 。 

BCTY365 网 上 社区 的 在 线 支 付 是 与 中 国 工商 银行 合作 来 共同 完成 的 。BCTY365 网 上 社区 的 在 线 
支付 操作 步骤 如 下 。 

(1) 登录 网 上 社区 ， 如 图 22.33 所 示 。 


四 您 当 前 的 位 置 >> 篇 程 司 奥 








名” 称 : VB 编程 词典 : YB 编程 词典 
所 属 版 本 :标准 版 专业 版 
价格 : 2,000.00 元 3 2,000.00 元 





版 。 权 : ”吉林 省 明日 科技 有 限 公司 


Gna] 


: ”吉林 省 明日 科技 有 限 公司 


(ma 





共有 编程 词典 2 个 每 页 显示 8 个 第 1 页 / 共 1 页 首页 上 一 页 下 一 页 尾 页 





22.33 ”在线 订购 的 操作 页 面 


(2) 购买 商品 。 在 本 页 中 ， 不 但 可 以 购买 商品 ， 还 可 以 查看 商品 的 详细 信息 和 购物 车 中 的 商品 信 
息 ， 如 图 22.34 所 示 。 


第 22 章 Apache+PHP+MySQL 实现 网 上 社区 





和 名” 称 : “7B 编程 词典 
|】 所 屋 版 本 :标准 版 
了 价格 : 2,000.00 元 
如 版 ”可 :吉林 省 明日 科技 有 限 公司 


ED Gr Er 








【IT 


Pa 


22.34 ”购买 商品 操作 页 面 


(3) 进入 购物 车 操作 页 面 。 在 该 页 面 中 ， 可 以 修改 购物 数量 ， 删 除 指定 商品 ， 清 空 购物 车 ， 继 续 


购物 和 统计 购买 商品 的 金额 ， 也 可 以 单 击 “ 结 算 ” 按 钮 进入 到 商品 结算 页 面 ， 如 图 22.35 所 示 。 








1 编程 词典 


商品 名 称 和 数量 (个 ) 操作 


2, 000.00 更 改 数量 | 删除 该 项 














《< ”继续 购买 ”清空 购物 车 > 商品 金额 总 计 : 2,000.00 元 








(4) 进入 到 购物 结算 页 面 ， 填 写 收 货 人 的 详细 信息 ， 确 认 后 提交 该 数据 ， 如 图 22.36 所 示 。 


图 22.35 购物 车 操作 页 面 





收 货 人 信息 


收 货 人 : 

性 别 : 

所 在 城市 : 
详细 联系 地 址 ; 
邮政 编码 : 

明 号 码 ; 


了 -mail : 


移动 电话 : 


站 请 务必 正确 填写 您 的 个 人 详细 信息 9 

PE 

Fa 可 

[RE 可 

Fi 和 | 
Fa 

Famm 

* 请 务必 正确 填写 悠 的 联系 地 址 和 邮编 ， 以 确保 订单 和 货物 顺利 达到 ? 


[e003 固定 电话 : sis4s7ss61 








站 请 选择 送 货 方式 
人 普通 邮 疗 
他 邮政 特快 专递 ENS 








生成 订单 











(5) 订单 确认 。 订 单 确认 以 后 ， 就 可 以 提交 订单 ， 准 备 进行 网 上 支付 ， 如 图 22.37 所 示 。 


22.36 填写 收 货 人 的 详细 信息 


中 
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吉林 省 明日 科技 有 限 公司 -编程 词典 订单 2007-12-13 11:54:39 
订单 号 : 07121311451945 
收 货 人 : 潘 林 邮编 : 130000 
移动 电话 : 1360433++#* 固定 电话 : 0431-84978981 
联系 地 址 : 长 春 市 
商品 名 称 单价 (元 ? 数量 小 计 ( 元 ) 
1 编程 局 典 2, 000.00 1 2, 000. 00 
商品 总 计 : 2,000.00 元 
邮费 : 240.10 元 


你 需要 支付 的 金额 总 计 为 2,240.10 元 
(i 
图 22.37 订单 确认 
(6) 进行 网 上 支付 。 在 这 里 可 以 选择 工行 网 上 支付 ， 也 可 以 选择 取消 该 订单 ， 如 图 22.38 所 示 。 


订单 号 : 07121311451945 需 支 付 金 额 ; 2240.1 元 

















* 只 有 在 网 上 支付 成 功 后 ， 






元 师 


* 我 们 会 














22.38 执行 网 上 支付 


接 下 来 的 操作 在 工行 B2C 支付 页 面 上 进行 。 首 先 网 上 社区 按照 工商 银行 B2C 订单 数据 规范 形成 提 
交 数 据 ， 并 使 用 工商 银行 提供 的 API 和 商户 证 书 对 订单 数据 签名 ， 形 成 form 表单 返回 客户 浏览 器 ， 表 
单 action 地 址 指向 工商 银行 接收 商户 B2C 订单 信息 的 servlet; 然后 在 客户 确认 使 用 工行 网 上 支付 后 ， 
提交 此 表单 到 工商 银行 ， 最 后 工行 网 银 系 统 接收 此 笔 B2C 订单 ， 对 订单 信息 和 商户 信息 进行 检查 ， 通 
过 检查 则 显示 工行 B2C 支付 页 面 。 

客户 通过 工行 B2C 支付 页 面 实现 网 上 支付 ， 商 户 查询 网 上 银行 的 账户 ， 如 果 货 款 已 经 到 账 ， 则 根 
据 客 户 指 定 的 方式 将 货物 送 达 客户 手中 。 

上 述 内 容 就 是 网 上 社区 系统 的 在 线 支付 流程 ， 涉 及 工商 银行 的 操作 内 容 这 里 不 做 讲解 。 这 里 主要 
讲解 一 下 如 何 将 订单 信息 提交 到 工商 银行 。 该 项 操作 主要 通过 shopping_tjdd.php 文件 来 实现 ， 首 先 从 
数据 库 中 读 取 订 单 信息 ， 然 后 将 订单 信息 进行 输出 ， 最 后 创建 “取消 订购 ”和 “工行 网 上 支付 ”两 个 
超 链接 ， 通 过 JavaScript 脚本 来 调用 不 同 的 执行 该 文件 。 其 关键 代码 如 下 。 

<?php ”include_once("conn/conn.php"); 。 ”include_once("top.php"); /连接 数据 库 和 网 站 的 头 文件 ?> 

<!-> 省 略 了 部 分 代码 <!--> 

<?php 

$ddnumber=base64_decode($_GET["ddno"]); // 对 获取 的 订单 编号 进行 base64 解码 

/获取 该 订单 的 金额 信息 

$sql=mysql_query("select * from tb_dd where ddnumber=". $ddnumber."",$conn); 

$info=mysql_fetch_array($sql); 








$amount=$info["totalprice"]; 
S$amount=str_replace(",","",number_format($amount,2)); /修改 数字 的 输出 格式 
Samount=str_replace(".",™",number_format($amount,2)); /修改 数字 的 输出 格式 
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?> 
过 // 省 略 部 分 HTML 代码 
<table width="630" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr><td width="159">&nbsp; 
<?php 
$sql=mysql_query("select totalprice from tb_dd where ddnumber=".base64_decode($_GET["ddno"]).""， 
$conn); 
$info=mysql_fetch_array($sql); 
echo "<font color=red><strong>".$info["totalprice"]."&nbsp; 元 </strong></font>"; 
?> </td> 
</tr> 
</table> 
“ // 省 略 部 分 HTML 代码 
<script language="javascript"> 
/打印 订单 
function openprintwindow(x,y,zj{ 
window.open("printwindow.php?ddno="+x+"&pv="+z, "newframe", "top=200,left=200,width=635,hei 
ght="+(230+20*y)+",menubar=no,location=no,toolbar=no,scrollbars=no,status=no"); 


</script> 
ee /省 略 部 分 HTML 代码 
<table width="630" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
一 二 


<td width="75"><img src="images/bg_14(14).jpg" width="69" height="20" style="cursor:hand" 
onclick="javascript:if(window.confirm(" 如 果 取 消 该 订单 ， 则 该 订单 将 被 删除 ， 您 需要 重新 购买 ! 
'")==true}{window.location.href='deletedd.php?ddno=<?php echo $_GET["ddno"];?>";}"/></td> 

< 一 执行 网 上 支付 一 -一 一 一 一 

<td width="125"><img src="images/bg_14(15)jpg" width="119" height="20" 
onclick="javascript:window.location.href='ddform.php?orderid=<?php echo 
base64_decode($_GET["ddno"]);?>&amount=<?php echo $amount;?>&orderDate=<?php echo 
date("Ymdhis");?>"," 
style="cursor:hand"/></td> 

</tr> 

</table> 
< 人 
<?php include_once("bottom.php"); /| 包含 网 站 的 尾 文件 ?> 


有 关 在 线 支付 流程 中 的 其 他 操作 实现 方式 已 经 在 22.10 节 中 进行 了 详细 的 讲解 , 这 里 不 再 袭 述 。 其 
具体 的 代码 还 可 以 参考 本 书 光盘 中 的 TM\02\bcty365\ 文 件 。 








22.11 社区 论坛 模块 设计 


社区 论坛 模块 为 网 站 的 浏览 者 提供 一 个 交流 的 平台 ， 以 此 来 扩大 网 站 的 影响 力 ， 汇 聚 更 多 的 人 气 ， 
宣传 企业 形象 ， 推 广 企业 产品 。 
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22.11.1 社区 论坛 模块 概述 


社区 论坛 模块 为 浏览 者 、 会 员 、 客 户 和 企业 之 间 提 供 一 个 大 的 交流 平台 ， 根 据 身份 的 不 同 ， 给 予 
其 分 别 拥有 不 同 的 操作 权限 ， 社 区 论坛 模块 的 操作 流程 如 图 22.39 所 示 。 














图 22.39 社区 论坛 流程 图 








路 寺 交 回 须 于 





在 本 论坛 中 ， 浏 览 者 只 能 够 查看 帖子 ， 注 册 会 员 既 可 以 查看 帖子 ， 也 可 以 发 布 和 回复 帖子 ， 管 理 


员 则 具有 发 布 、 回 复 、 查 看 和 删除 的 权限 。 


22.11.2 ”社区 论坛 模块 技术 分 析 


在 社区 论坛 模块 的 实现 过 程 中 , 通过 JavaScript 脚本 和 下 拉 列 表 框 的 结合 实现 一 个 不 同 版 块 之 间 快 
速 跳 转 的 功能 ， 从 而 能 够 更 加 灵活 、 方 便 地 实现 不 同 版 块 之 间 的 跳 转 。 
下 面 分 析 该 技术 是 如 何 实现 的 ， 该 技术 的 实现 综合 3 个 方面 的 内 容 ， 以 一 个 下 拉 列 表 框 为 主 ， 通 





过 PHP 语句 从 数据 库 中 读 取 数据 作为 下 拉 列 表 框 的 值 ， 应 

















onchange 事件 来 调用 JavaScript 脚本 ， 实 


现 不 同 版 块 之 间 的 跳 转 。 这 里 以 bbs_top .php 文件 中 的 快速 跳 转 功能 为 例 进行 分 析 。 其 关键 代码 如 下 。 


代码 位 置 : 光盘 \TM\sl\22\bcty365\bbs_top.php 


< 上 -创建 一 个 下 拉 列 表 框 ， 指 定名 称 为 select_type， 并 且 设 置 其 属性 ， 通 过 onchange 事件 来 调用 JavaScript 脚 


本 文件 ， 实 现 页 面 跳 转 --> 
<select name="select_type" class="inputcss" 


onChange="javascript:window.location=this.options[this.selectedindex].value;" > 


<?php 


// 通 过 PHP 语句 从 数据 库 中 读 取 数据 ， 使 用 数据 的 ID 作为 下 拉 列 表 框 的 值 ， 使 用 数据 的 标题 ttle 作为 下 拉 列 表 框 显 


示 的 内 容 


$sql=mysql_query("select * from tb_type_small order by createtime desc",$conn); 
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$info=mysql_fetch_array($sql); 
if($info=="™"){ 
echo "<option> 暂 无 讨论 区 </option>"; 
}else{ 
echo "<option>- 版 块 快速 跳 转 -</option>"; 
dof /应 用 do…while 循环 语句 输出 下 拉 列 表 框 中 的 值 
echo "<option value="'bbs_list.php?id=".$info[id]."">".$infoltitle]."</option>"; 
} 
while($info=mysql_fetch_array($sql)); /应 用 do.…while 循环 语句 结束 
} 
?> 
</select> 


该 技术 实现 的 运行 结果 如 图 22.40 所 示 ， 它 将 实现 从 硬件 查询 模块 跳 转 到 PHP 模块 。 
四 当前 位 置 >》 社 区 论坛 论坛 版 块 查找 帖子 星 


计算 机 硬件 冰 询 版 块 
意 | 心情 主 是 


| ”同日 我 的 计算 机 不 能 启动 了 ? 












共有 帖子 ! 条 每 页 显示 30 条 第 ! 页 / 共 1 页 | Software engineering/managen 


sc 
indors 2003 
人 和 


asp/asp. nat 


22.40 ”版 块 跳 转 功 能 的 运行 结果 


22.11.3 ”论坛 分 类 的 实现 过 程 


论坛 分 类 可 以 分 为 两 类 ， 一 个 是 论坛 中 大 的 版 块 分 区 ， 分 为 6 个 版 块 : 综合 信息 讨论 区 、 操 作 系 


统 、 程 序 设计 交流 区 、 网 管 技术 应 用 、Web 程序 开发 和 数据 库 技术 ， 其 数据 存储 于 tb_type_big 表 中 。 
另 一 个 是 对 应 不 同 的 版 块 中 不 同 语言 和 技术 的 分 类 ， 分 为 11 种 ， 其 数据 存储 于 tb_type_small 表 中 。 论 
坛 分 类 的 运行 结果 如 图 22.41 所 示 。 


eb 程序 开发 太 梧 得 。 | 热门 由 要 
计算 机 硬件 和 咨询 

创建 时 间 : 2009-12-05 00:00: 四 和。 
操作 系 六 用 则 小 。 | 热门 帖 下 
asp/ asp net 

创建 村 间 : 2000-12-16 00:00: 四 


De 
1 
他 时 间 : 2009-12-05 00:00:m 
CEEEEEEEEB 
苦 合 信息 讨论 区 


lt : 2005-12- 
戎 件 工 竹 / 管 理 | se; 


6 o 
他 时 间 : 2006-12-05 00:00: 四 








ngineeriagfaanagement 


图 22.41 论坛 分 类 的 运行 结果 
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论坛 分 类 的 实现 原理 很 简单 ， 首 先 从 tb_type_big 表 中 读 取 6 个 版 块 中 的 数据 ， 进 行 循环 输出 ， 然 




















后 在 版 块 中 嵌 套 循环 ， 用 于 输出 不 用 语言 的 分 类 数据 。 该 功能 主要 通过 bbs_index.php 文件 来 完成 ， 
bbs index.php 文件 的 程序 代码 如 下 。 
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代码 位 置 ， 光盘 \mr\sl\22\bcty365\bbs_index.php 


<?php include_once("top.php"); // 调 用 网 站 头 文件 
include_once("bbs_top.php"); // 调 用 社区 论坛 的 头 文件 

J 

<?php 


/循环 输出 数据 表 tb_type_big 中 的 6 个 版 块 数据 
$sql=mysql_query("select * from tb_type_big order by createtime desc",$conn); 
$info=mysql_fetch_array($sql); 


if($info==falseX{ // 如 果 返 回 值 为 false， 则 执行 下 面 的 内 容 

yp 

天 /省略 了 部 分 代码 

<?php 

}else{ // 如 果 返 回 值 为 rue， 则 执行 do…while 循环 语句 

上 外 部 嵌 套 循环 ， 输 出 论坛 中 的 版 块 分 类 数据 */ 
dof 

?> 

… 省 略 了 部 分 代码 


<table width="750" border="0" align="center" cellpadding="0" cellspacing="1" bordercolor="#FFFFFF" 
bgcolor="#6EBEC7"> 
<?php 
/循环 输出 tb_type_small 表 中 的 不 同 语言 和 技术 的 分 类 数据 
$sql1=mysql_query("select * from tb_type_small where bigtypeid=".$info["id"]."",$conn); 
$info1=mysql_fetch_array($sql1); 


if($info1==falseX{ // 判 断 如 果 返 回 值 为 false， 则 执行 下 面 的 内 容 
?> 
… 省 略 了 部 分 代码 
<?php 
}else{ // 如 果 返 回 值 为 true， 则 执行 下 面 的 内 容 ， 输 出 该 版 块 中 对 应 语言 和 技术 的 帖子 详细 信息 
Ee 
// 省 略 了 部 分 代码 
<?php ”上 * 内 部 嵌 套 循环 ， 输 出 不 同 语言 和 技术 的 分 类 */ 
do{ ?> 
<tr> 
<td height="30"><font color="#666666"> 创 建 时 间 : <?php echo S$info1["createtime"];?></font> 
</td> 
</tr> 
省略 了 部 分 代码 
<?php 
}while($info1=mysql_fetch_array($sql1)); 
六 内 部 嵌 套 循环 结束 “*/ 
} 
?> 
</table> 
<?php 


jwhile($info=mysql_fetch_array($sql)); 
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上 外 部 嵌 套 循环 结束 ， 对 版 块 中 的 大 类 进行 输出 7 


和 
<?php include_once("bottom.php"); ?> 
et 
在 应 用 do.…while 循环 语句 时 ，while 后 的 分 号 不 能 省 略 。 


22.11.4 “论坛 帖子 浏览 的 实现 过 程 


论坛 帖子 浏览 主要 输出 指定 帖子 的 详细 信息 ， 包 括 发 帖 人 、 用 户 级 别 和 注册 的 时 间 ， 以 及 帖子 的 
主题 、 内 容 和 发 帖 时 间 ， 包 括 上 传 的 图 片 。 本 模块 中 是 用 户 权 限 使 用 体现 得 最 明显 的 地 方 ， 可 以 分 为 3 
种 情况 : 第 一 以 浏览 者 进行 登录 ， 只 能 是 浏览 帖子 的 内 容 ， 没 有 其 他 权限 ; 第 二 以 会 员 进行 登录 ， 可 
以 对 帖子 进行 回复 ， 发 表 自己 的 看 法 ;第 三 以 管理 员 的 身份 进行 登录 ， 不 但 可 以 回复 帖子 ， 而 且 可 以 
对 任何 人 发 布 和 回复 的 帖子 进行 删除 和 项 帖 的 操作 。 下 面 就 来 看 一 下 以 管理 员 身 份 进行 登录 时 都 具备 
哪些 权限 ， 运 行 结果 如 图 22.42 所 示 。 

ET 


OO 2007-12-12 13:19:47 
稚 想 请 教 一 下 如 何 配置 PE 的 环境 


发 贴 人 : lx 
信箱 仿 0ica 易 Tf 地 址 
用 户 级 别 : 管理 员 


发 巾 总数 :5 ie 


注册 时 间 : 2007-12-01 13:15:25 


从 OO) 2007-12-12 13:21:48 山中 


发 贴 人 : lx 回复 主题 : PHP 的 环境 配置 方法 
信箱 国 oica 蛙 I 地 址 





用 户 级 别 : 管理 员 
发 贴 总 数 : 5 详细 内 容 请 参考 PIPE 数据 库 系 统 开发 完全 手册 中 的 第 1 章 
注册 时 间 : 2007-12-01 13:15:25 





图 22.42 管理 员 浏览 帖子 的 结果 
论坛 帖子 浏览 的 功能 通过 bbs_ looksphp 文件 完成 ， 首 先 根据 传递 的 ID 值 读 取 指 定 的 帖子 数据 ， 
然后 判断 登录 用 户 的 类 型 ， 最 后 根据 用 户 不 同 的 类 型 执行 不 同 的 操作 。 其 代码 如 下 。 


代码 位 置 : 光盘 \TMN\sl\22\bcty365\bbs_lookbbs.php 
<?php 
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/根据 $_GET 传递 的 数据 获取 tb_bbs 中 的 数据 
$sqlb=mysql_query("select * from tb_bbs where id=".$_GETT["id"]."",$conn); 
$infob=mysql_fetch_array($sqlb); 
/根据 $_GET 传递 的 数据 获取 tb_user 中 的 数据 
$sql4=mysql_query("select * from tb_user where id=".$infob["userid"]."",$conn); 
$info4=mysql_fetch_array($sql4); 
?> 
本 /省略 了 部 分 HTML 代码 
<table width="180" border="0" align="center "cellpadding="0" cellspacing="0"> 
<tr> 
<td height="22"> 用 户 级 别 : 
<?php 
/根据 用 户 信息 表 tb_user 中 字段 usertype 的 值 判断 该 用 户 的 类 型 
// 如 果 值 为 1 则 是 管理 员 ， 值 为 2 则 是 后 台 管理 员 ， 值 为 0 则 是 普通 会 员 
if($info4["usertype"]=="1") echo "管理 员 ";else echo "普通 会 员 "; 
全 
</td> 
</tr> 
</table> 
es // 省 略 了 部 分 HTML 代码 
<table width="500" height="200" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="35" height="20"><div align="center"><img src="images/lt_15(11).jpg" width="25" 
height="25"></div></td> 
<td><?php echo S$infob["createtime"];?></td> 
</tr> 
<tr> 
<td height="150" colspan="2"> 
<?php 
/判断 tb_bbs 表 中 的 字段 photo 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 
if($infob[photo]!="")}{ 


$photos=substr($infob[photo],2,70); // 获 取 图 片 在 服务 器 中 的 存储 路 径 
echo (stripslashes($infob["content"])); // 输 出 帖子 的 内 容 
echo "<img src=\"$photos\">"; /根据 获取 的 图 片 路 径 ， 输 出 服务 器 中 的 图 片 
}else{ /I 如 果 tb_bbs 表 中 的 图 片 字段 photo 为 空 ， 则 执行 下 面 的 内 容 
echo (stripslashes($infob["content"])); // 只 输出 帖子 的 内 容 
?> 
</td> 
</tr> 
</table> 
<table width="530" height="20" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 


<td width="239" height="30">&nbsp;</td> 
<td width="291"><img src="images/lt_15(5).jpg" width="72" height="23" style="cursor:hand" onclick=" 
<?php 
/如 果 $_SESSION["unc"] 的 值 为 空 ， 则 不 可 以 进行 项 帖子 的 操作 
if$_SESSIONT"unc']=="")f 
echo "javascript:alert(' 请 先 登录 本 站 ， 然 后 进行 此 操作 ! ');window.location.href= 
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index.php' 
Jelse{ 
/否则 将 判断 当前 用 户 的 类 型 ， 如 果 是 管理 员 则 可 以 项 帖 
$sqlu=mysql_query("select usertype from tb_user where 
usernc=".$_SESSION["unc"]."",$conn); 
$infou=mysql_fetch_array($sqlu); 


if($infou["usertype"]==1){ // 如 果 用 户 的 类 型 为 1， 则 有 项 帖 的 权限 
echo "javascript:window.location.href="'settop.php?id=".$infob["id"]."™"; 
jelse{ // 否 则 不 具备 该 权限 
echo "javascript:alert( 对 不 起 ， 您 不 具备 该 操作 权限 ! ");"; 
} 
} 
了 > 
">&nbspi&nbsp; 
<?php 


/判断 当前 用 户 是 否 具有 删除 帖子 的 权限 
if($_SESSION["unc"]!="™")}{ 
// 条 件 为 用 户 不 能 为 空 ， 并 且 是 管理 员 ， 才 具备 删除 帖子 的 权限 
$sqlu=mysql_query("select usertype from tb_user where usernc="".$_SESSION 
["unc"]."",$conn); 
$infou=mysql_fetch_array($sqlu); 
if($infou["usertype"]==1){ 
?> 
<img src="images/lt_15(10)jpg" onclick="javascriptif(window.confirm(' 您 确定 删除 该 帖 吗 ? 
")==true){window.location.href='bbs_delete.php?id=<?php echo S$infob["id"]?>";}" style="cursor:hand"/> 


<?php 
} 
} 
?> 
</td> 
</tr> 
</table> 
yg 

6 涪 明 


上 面 给 出 的 是 该 文件 的 部 分 代码 , 主要 讲解 了 该 功能 的 实现 方法 , 完整 的 代码 可 以 参考 本 书 光 
盘 中 的 TM\02\bety365\bbs lookbbs.php 文件 。 


22.11.5 ”论坛 帖子 发 布 的 实现 过 程 


论坛 帖子 发 布 通 过 两 个 文件 来 完成 ， 一 个 是 帖子 发 布 信息 的 提交 页 bbs_pubs.php， 另 一 个 是 对 提 
交 的 数据 进行 处 理 的 retrieve.php 文件 。 该 功能 实现 的 运行 结果 如 图 22.43 所 示 。 


MySQL 从 入 门 到 精通 





CT 噩 | 





帖子 主题 : 
从 对 您 现在 的 心情 如何? 
会 @r re@r° er 
当前 用 户 : 1x 辣 c 号 Cc 鲁 c 国 “ 多] c 三 
信箱 仿 0ica 易 芽 地 址 着 CCArrBr @ 


用 户 级 别 : 管理 员 四 片 : | (图 片 不 能 #32, 格式 为 st/ jps) 


发 贴 总 数 : 5 
注册 时 间 : 2007-12-01 13;15:25 


EE 


图 22.43 ”帖子 发 布 模块 的 运行 结果 


在 发 布 信息 的 提交 页 中 ， 显 示 当 前 用 户 的 个 人 信息 ， 设 置 添加 数据 表单 元 素 ， 其 中 表单 元 素 的 设 
计 如 表 22.3 所 示 。 
表 22.3 发 布 信息 页 中 使 用 的 表单 元 素 























名 称 | 元 素 类 型 重要 属性 含义 
form bbs form method="post" action="retrieve.php" enctype="mulf form-data” 发 帖 表 单 
class="inputcss" style="background-color#6EBEC7"> 
<2php 
$sql=mysql_query("select * 位 om tb_type_small order by createtime 
desc",$conn): 
Sinfo=mysql _fetch_array($sql): 
a 选择 发 表 
echo "<option> 暂 无 讨论 区 </option>"; . 
bbs_type select jelse{ 0 的 民 
dat 言 或 者 技 
a 术 的 类 别 
<option value="<?php echo $info[id] :?>"<?php 这 $_GET[id] 一 $info[id]) 
{echo "selected=\"selected\"":}?>><?php echo $infoftitle]:?></option> 
<Iphp  } 
while($info=mysql fetch_array($sqD): 
j2> 
bbs title text class="inputcss" style="background-color-#6EBECT7"> 帖子 标题 
bbs head radio value="<?php echo("images/bbsface/face".($i-1).".gif"):?>" 表情 图 
bbs photo file id="bbs_photo" class="inputcss" style= "background-color#6EBEC7" /> 上 传 图 片 
contentl textarea id="contentl" class="inputcss" style="backeround-color.#6EBEC7"> 帖子 内 容 
Submit submit Value=" 提 交 " 提交 表单 
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在 retrieve.php 页 中 对 表单 提交 的 数据 进行 处 理 ， 将 数据 存储 到 tb_bbs 表 中 ,并 且 更 新 用 户 信息 表 








tb_user 中 pubtimes 字段 的 值 ， 其 中 还 应 用 了 图 片上 传 技 术 ， 将 图 片上 传 到 服务 器 中 指定 的 文件 夹 下 。 
retrieve.php 文件 的 代码 如 下 。 


代码 位 置 ， 光盘 \TMN\s122\bcty365\retrieve.php 
<?php session_start(); /| 初始 化 session 变量 
$title=$_POST[bbs_title]; // 获 取 帖 子 的 标题 
$content=$_POST[content1]; // 获 取 帖 子 的 内 容 
上 判断 提交 的 帖子 主题 和 帖子 内 容 是 否 为 空 */ 
if($title=="){ 

echo "<script>alert(' 请 输入 帖子 主题 ");history.back();</script>"; 

exit; } 
读 $content==""){ 

echo "<script>alert(' 请 输入 帖子 内 容 !");history.back();</script>"; 

exit; } 
ey 
include_once("conn/conn.php"); /| 连接 数据 库 
/根据 $_SESSION["unc"] 的 值 读 取 数 据 库 中 用 户 的 信息 
$sql=mysql_query("select * from tb_user where usernc=".$_SESSION["unc"]."",$conn); 


$info=mysql_fetch_array($sql); /检索 指定 条 件 的 数据 信息 
$userid=$info[id]; // 获 取 用 户 id 
$typeid=$_POST[bbs_type]; // 接 收 版 块 名 称 
Stitle=$_POST[bbs title]; /接收 帖子 主题 
$content=$_POST[content1]; /接收 帖子 内 容 
$head=$_POST[bbs_head]; /接收 头像 
$createtime=date("Y-m-j H:i:s"); // 获 取 系 统 当前 时 间 
$lastreplytime=$createtime; /将 当前 时 间 赋 给 变量 


Sreadtimes=0; 
Slink=date("YmjHis"); 
if($_FILES['bbs_photo"]["name"]==true}{ // 上 传 图 片 ， 判 断 文件 是 否 存 在 ， 如 果 存 在 则 执行 下 面 的 内 容 
$photo_name=strtolower(stristr($_FILES["bbs_photo"]["name"],"."));// 获 取 图 片 后 缀 名 , 将 字符 转 成 小 写 
if($photo_name!=".gif" & $photo_name!=".jpg" & $photo_name!=".jpeg" ){ // 判 断 图 片 的 格式 是 否 符合 要 求 
echo "<script>alert(' 您 上 传 的 图 片 格式 不 正确 !");history.back();</script>"; 


Jelse{ 

$paths1=$link.mt_rand(1000000,9999999).$photo_name; /创建 图 片 的 名 称 
$photos="./upfile/".$paths1; /创建 图 片 的 存储 路 径 

move_uploaded file($_FILES[bbs_photo']["tmp_name"],$photos); 
/将 图 片 存储 到 指定 的 文件 夹 下 
/向 数据 库 添加 数据 
if(mysql_query("insert into tb_bbs(userid,typeid,title,content,createtime,lastreplytime,head,readtimes, 
top,photo) 


values(".$userid.",".$typeid.",". $title.™",".$content.",".$createtime.",".$lastreplytime.",".$head.",".$readtimes 
.",'0','$photos')", $conn)\{ 
mysql_query("update tb_user set pubtimes=pubtimes+1",$conn); 
// 更 新 tb_user 中 pubtimes 字段 的 值 
echo "<script>alert( 新 帖 发 表 成 功 !"):history.back();</script>"; 
}else{ 
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echo "<script>alert(' 新 帖 发 表 失 败 !);history.back();</script>"; 
} 


} 
}else{// 如 果 没 有 提交 图 片 ， 则 执行 下 面 的 内 容 
if(mysql_query("insert into tb_bbs(userid,typeid ,title,content,createtime,lastreplytime,head,readtimes,top) 
values 
(".$userid.",".$typeid."," .$title.",”".$content.”",".$createtime.",".$lastreplytime.",”".$head.”",".$readtimes.",'0")", 
$conn)){ 
mysql_query("update tb_user set pubtimes=pubtimes+1",$conn); 
echo "<script>alert( 新 帖 发 表 成 功 ");history.back();</script>"; 
Jelse{ 
echo "<script>alert( 新 帖 发 表 失 败 !");history.back();</script>"; 
} 


22.11.6 ”论坛 帖子 回复 的 实现 过 程 


回复 论坛 中 的 帖子 ， 必 须 以 会 员 或 者 管理 员 的 身份 进行 登录 ， 否 则 不 能 进行 帖子 的 回复 操作 ， 其 
运行 结果 如 图 22.44 所 示 。 


饮 ?IF 篇 程 词典 ,程序 开发 测试 


9 





O ao0r-l1-30 :2 





发 贴 人 : TSoft 
信箱 七 oicq 易 IT 地 址 
户 令 别 : 普通 会 员 
二 Re 欢迎 大 家 提出 宝 货 意 见 | 
注册 时 间 : 2007-11-18 08:42:43 Ca 
9 回复 主题 : 砍 二 和 模 天 测 芭 
图 片 : 民 :\ersele\webpegever\bety 加 | 图片 不 能 要 过 2Wp, 格式 为 . gif/ jp 上 
发 贴 人 : TSoft 
信箱 仿 0icq 易 I 地 址 论坛 回复 模块 测试 ,程序 测试 中 应 该 注意 的 问题 司 
用 户 级 别 : 普通 会 员 
发 贴 总 数 : 4 回 
注册 时 间 : 2007-11-18 08:42:43 提交 | 重 填 


图 22.44 ”论坛 帖子 回复 的 运行 结果 


论坛 帖子 回复 功能 的 实现 主要 通过 bbs_looks.php 和 savereply.php 两 个 文件 。 其 中 应 用 JavaScript 
脚本 对 回复 帖子 的 文本 框 进行 输出 和 隐藏 的 控制 。 在 bbs_looks.php 文件 中 ， 帖 子 回复 使 用 的 表单 元 素 
如 表 22.4 所 示 。 
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表 22.4 论坛 帖子 回复 中 的 重要 表单 元 素 


























form reply | form method="post" action="savereply.php" enctype="multipart/form-data"> | 回复 表单 
reply title | text | class="inputcss" id="reply title” | 回复 帖子 主题 
bbsid hidden | value="<?php echo $infob["id"]:2>" | 对 应 帖子 的 ID 
bbs head Tadio | value="<?php echo("images/bbsface/face".($i-1).".gif"):?>" | 表情 图 

bbs photo | file | id="bbs_photo" class="inputcss" | 上 传 图 片 
contentl textarea | id="content1" | 回复 帖子 内 容 
Submit submit Value=" 提 交 " 提交 表单 








在 帖子 回复 表单 bbs_ looks.php 页 中 ， 首 先 判断 登录 用 户 是 否 具 有 回复 的 权限 ， 然 后 根据 提交 的 值 
展开 回复 表单 的 文本 框 ， 在 文本 框 中 输入 回复 的 主题 和 内 容 ， 最 后 将 数据 提交 到 表单 处 理 页 
savereply.php 中 。bbs_looks.php 的 主要 代码 如 下 。 

代码 位 置 ， 光盘 \TMVsI\22\bcty36S\bbs_ lookbbs .php 

<script language="javascript"> 


























/设计 回复 帖子 表格 的 输出 方式 
function show_reply(X{ /定义 一 个 函数 
ifreply_bbs1.style.display=="" /判断 当 display 的 值 为 空 时 
reply_bbs1.style.display="none" ; // 则 输出 表格 
button_show_bbs.value=" 回 复 帖 子 "; // 显 示 回 复 帖 子 
) 
else if(reply_bbs1.style.display=="none") /| 判断 当 display 的 值 为 none 时 
reply_bbs1.style.display="™" ; // 则 不 输出 表格 
button_show_bbs.value=" 关 闭 窗口 "; // 显 示 关 闭 窗 口 
} 
下 
</script> 
< 一 一 一 一 一 一 一 一 一 一 判断 登录 用 户 是 否 具有 回复 的 权限 一 一 一 一 一 一 一 一 一 一 一 一 一 
<img src="images/lt_15(9).jpg" width="72" height="23" id="button_show_bbs" style="cursor:hand" onClick=" 
<?php 


if($_SESSION["unc"]==" 
echo "javascript:alert(' 请 先 登录 本 站 ， 然 后 回复 帖子 !);window.location.href='index.php";"; 


Jelse{ 
?> 
show_reply() 
<?php } > Ns 


表单 处 理 页 savereply.php 将 表单 提交 的 数据 存储 到 指定 的 数据 库 中 ， 其 实现 的 方法 与 论坛 发 布 中 
的 表单 处 理 技术 是 相同 的 ， 有 关 该 技术 的 详细 讲解 请 参考 22.11.5 节 ， 这 里 不 再 獒 述 。 
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22.12 后 台 首 页 设计 


作为 一 个 完整 的 网 上 社区 系统 ， 要 想 能 够 及 时 地 对 网 站 进行 管理 和 维护 ， 必 须 具 有 一 个 强大 的 后 








台 管 理 系统 ， 对 网 上 社区 系统 中 的 数据 进行 更 新 和 维护 。 
22.12.1 后 台 首 页 概述 


网 上 社区 系统 的 后 台 管理 采用 的 是 一 种 简单 的 框架 结构 ， 通 过 switch 语句 来 实现 。 其 具体 内 容 


如 下 。 


(1) 软件 试用 管理 : 
(2) 编程 词典 管理 : 
(3) 在 线 订购 管理 : 
(4) 软件 升级 管理 : 
(5) 站 内 公告 管理 : 
(6) 技术 支持 管理 : 


包括 软件 试用 产品 的 添加 和 删除 。 

包括 编程 词典 版 本 的 添加 、 删 除 和 编程 词典 内 容 的 添加 和 删除 。 

主要 用 于 管理 用 户 提交 的 订单 。 

包括 升级 包 的 添加 、 删 除 和 序列 号 的 添加 和 删除 。 

主要 用 于 添加 和 删除 站 内 公告 。 

主要 用 于 添加 常见 问题 和 删除 常见 问题 ， 以 及 对 客户 反馈 信息 进行 管理 。 


下 面 看 一 下 本 案例 中 提供 的 后 台 首 页 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\02\bcty365\admin\index.php， 
如 图 22.45 所 示 。 





日 编程 词典 管理 


刻 加 编程 局 典 版 本 编辑 编程 词典 版本 
A hi 


尖 加 编程 同 典 
编辑 编程 同和 


田 技术 支持 管理 
曙 软件 试用 管理 
田 软件 升级 管理 


田 在 线 订购 管理 











2007-12-08 11:52:34 





2007-12-08 11:52:30 





2007-12-06 11:51:46 

















2007-12-06 11:51:17 











客户 服务 中 心 热 线 : 0431-84978981 Copyriaht 2007 红 ) 吉林 省 明日 科技 有 限 公司 





图 22.45 BCTY365 网 上 社区 系统 后 台 首 页 
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22.12.2 ”后台 首页 技术 分 析 





网 上 社区 后 台 首 页 的 设计 主要 应 用 switch 语句 和 include 语句 , 其 实现 的 原理 是 : 应 用 switch 语句 ， 
根据 超 链 接 中 传递 的 变量 值 进 行 判 断 ， 根 据 不 同 的 变量 值 应 用 include 调用 不 同 的 子 文件 。 该 技术 的 实 
现 流 程 如 图 22.46 所 示 。 


ES 


| 
| 
| 
| 











人 硼 )J| 应 用 include 根 据 不 同 的 | 
值 输 出 不 同 的 子 文件 | 


0431-84978981 Copyrizht 2007 代 ) 吉林 省 明日 科技 








图 22.46 网 上 社区 后 台 首 页 设计 流程 
为 能 够 更 好 地 理解 这 个 技术 ， 先 来 了 解 一 下 switch 语句 。 该 语句 的 格式 如 下 。 


switch( expr X{ llexpr 条 件 为 变量 名 称 
case expr1: llcase 后 的 expr1 为 变量 的 值 
statement1; /冒号 ":" 后 的 是 符合 该 条 件 时 要 执行 的 部 分 
break ; /应 用 break 来 跳 离 循环 体 


| 


参数 expr 是 表达 式 的 值 ， 即 switch 语句 的 条 件 变量 的 名 称 ; 参数 exprl 放置 于 case 语句 之 后 ， 是 
要 与 条 件 变 量 expr 进行 匹配 的 值 中 的 一 个 ，statementl 是 在 参数 exprl 的 值 与 条 件 变量 expr 的 值 相 匹 
配 时 执行 的 代码 ;break 语句 实现 终止 语句 的 执行 ， 即 当 语 句 在 执行 过 程 中 ， 遇 到 break 就 停止 执行 ， 
跳出 循环 体 ; default 是 case 的 一 个 特例 ， 匹 配 了 任何 其 他 case 都 不 匹配 的 情况 ， 并 且 是 最 后 一 条 case 
语句 。 

通过 switch 和 include 语句 来 实现 后 台 管 理 功能 的 设计 是 一 个 很 好 的 方法 ， 不 但 实现 过 程 简单 ， 而 
且 操 作 也 非常 灵活 。 其 关键 代码 如 下 : 


a 
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代码 位 置 : 光盘 \TM\sl22\bcty365\ admin\wzdh php 


<?php 
switch($htgl){ /根据 变量 提交 的 不 同 值 
case "添加 编程 词典 版 本 ": /判断 与 变量 提交 的 值 是 否 相 同 
include("addbb.php"); // 如 果 值 相同 ， 则 调用 指定 的 文件 
break: // 并 且 跳 出 本 次 循环 
case "编辑 编程 词典 版 本 ": 
include("editbd.php"); 
break; 
se // 部 分 代码 省 略 
case ™: // 当 变量 的 值 为 空 时 
include("edittell.php"); // 调 用 该 文件 
break; 
也 
2 


22.12.3 ”后 台 首页 的 实现 过 程 


在 后 台 首页 的 设计 过 程 中 ， 以 switch 循环 语句 为 基础 ， 架 设 整 个 后 台 管 理 功能 的 框架 结构 ， 充 分 
发 挥 include 语句 的 作用 ， 调 用 不 同 的 文件 执行 不 同 的 管理 操作 ;应 用 JavaScript 脚本 来 控制 栏目 列表 
的 输出 和 隐藏 。 

控制 栏目 列表 的 输出 和 隐藏 在 menu.php 文件 中 进行 ， 首 先 定义 一 个 函数 changeO 用 于 控制 表格 的 
输出 和 隐藏 ， 然 后 在 表格 中 应 用 onclick 事件 传递 不 同 的 值 到 自 定义 函数 change0， 最 后 根据 不 同 的 值 
显示 不 同 的 内 容 。 其 关键 代码 如 下 。 

代码 位 置 : 光盘 \TM\sl22\bcty365\adminmenu.php 


<script language="javascript"> 
// 通 过 脚本 语言 控制 文本 框 的 伸展 和 收缩 


function change(x,y}{ /定义 一 个 函数 
说 x.style.display=="none"){ /| 判断 当 样式 的 值 为 none 时 
x.style.display=""; // 判 断 当 样式 的 值 为 none 时 ， 输 出 样式 的 值 为 空 
} 
else if(x.style.display=="™"){ /判断 当 样 式 的 值 为 空 时 
x.style.display="none"; /判断 当 样 式 的 值 为 空 时 ， 输 出 样式 的 值 为 none 
y.background="images/bg_16_11.jpg"; // 输 出 背景 图 片 
8 
} 
</script> 


<table width="175" height="28" border="0" align="center" cellpadding="0" cellspacing="4" 
onclick="change(tz1,img_tz1)" style="cursor:hand"> 
<tr> 
<td background="images/bg_16_11.jpg" id="img_tz1" class="a4"><div align="left"><img 
src="images/bg_16_21.jpg">&nbsp; 编 程 词典 管理 </div></td> 
</tr> 
</table> 
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<table name="tz1" id="tz1" width="170" height="40" border="0" align="center "cellpadding="0" cellspacing="0" 
<?php 

// 根 据 变量 的 值 选 择 执行 的 内 容 ， 当 变量 的 值 不 为 真 时 ， 隐 藏 该 表格 

if(($_GET[htgl]==" 添 加 编程 词典 版 本 " ||$_GET[htgl]==" 编 辑 编程 词典 版 本 " 上 |$_GETIhtgl]==" 添 加 编程 词典 " 
ll$_GETIhtg]==" 编 辑 编程 词典 " )/ 


2> 

style="display:none”" 
<?php } ?> 
> 


<tr> 
<td width="40" height="24" background="images/bg_16_16.jpg">&nbsp;</td> 
<td width="114" background="images/bg_16_16.jpg"><div align="left"><a href="default.php?htgl= 添 加 编程 
词典 版 本 "> 添加 编程 词典 版 本 </a></div></td> 
</tr> 
</table> 


/ 
这 里 给 出 的 只 是 后 台 首 页 实现 过 程 中 的 主要 代码 ， 详 细 代 码 可 参考 本 书 的 光盘 
TM\02\bcty365\admin\ 文 件 夹 下 的 相关 文件 。 


22.13 ”编程 词典 管理 模块 设计 


本 模块 的 功能 是 对 网 站 中 的 编程 词典 进行 管理 ， 包 括 添加 编程 词典 版 本 、 编 辑 编程 词典 的 版 本 、 
添加 编程 词典 和 编辑 编程 词典 。 


22.13.1 编程 词典 管理 模块 概述 


本 模块 的 主要 功能 是 管理 网 站 中 在 线 出 售 的 编程 词典 软件 ， 实 现 对 编程 词典 软件 及 时 的 更 新 和 维 
护 ， 其 管理 的 内 容 主要 包括 添加 和 编辑 编程 词典 的 版 本 ， 添 加 和 编辑 编程 词典 的 详细 信息 。 在 添加 编 
程 词典 时 ， 包 括 名 称 、 版 权 、 图 片 、 类 别 、 内 容 简介 和 不 同 版 本 的 共同 点 ; 编辑 编程 词典 包括 版 本 、 
价格 、 简 介 、 功 能 和 服务 ， 其 中 每 一 个 编程 词典 软件 只 可 以 编辑 一 次 ， 不 可 以 进行 重复 编辑 ， 如 果 要 
重新 编辑 ， 就 必须 将 已 经 编辑 过 的 信息 删除 。 








22.13.2 ”编程 词典 管理 模块 技术 分 析 





在 编程 词典 管理 模块 中 ， 应 用 到 图 片上 传 技术 ， 通 过 该 技术 将 编程 词典 的 界面 效果 上 传 到 服务 器 
的 指定 文件 夹 下 。 该 技术 主要 通过 move_uploaded file0 函 数 来 实现 ， 其 中 还 应 用 到 is_dir0、mkdir0 函 
数 ， 判 断 指 定 的 文件 夹 是 否 存 在 和 创建 文件 夹 ， 还 有 mt rand0、strstr0 函 数 和 $_FILES[] 全 局 变量 。 为 
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了 更 好 地 理解 和 掌握 图 片上 传 处 理 技术 ， 这 里 以 编程 词典 模块 





的 savebccd php 文件 为 例 进行 讲解 。 


首先 应 用 is_dir0 函 数 判断 在 服务 器 中 是 否 存在 指定 的 文件 夹 ， 如 果 不 存在 ， 则 应 用 mkdir0 函 数 创 


建 一 个 新 的 文件 夹 。 


然后 应 用 $_FILES[] 全 局 变量 获取 图 片 名 ， 应 用 strstr0 函 数 获取 图 片 文件 的 后 缀 名 ， 为 避免 出 现 同 
名 文件 覆盖 ， 这 里 应 用 系统 的 当前 时 间 和 mtrandO 函 数 获取 的 一 个 7 位 随机 数字 作为 图 片 的 名 称 。 

最 后 确定 图 片 在 服务 器 中 存储 的 路 径 ， 将 图 片上 传 到 指定 的 文件 夹 下 。 而 数据 库 中 存储 的 数据 是 
图 片 在 服务 器 中 的 路 径 ， 当 需要 输出 图 片 时 ， 只 需要 获取 到 数据 库 中 图 片 的 路 径 即 可 。savebccdphp 





文件 的 代码 如 下 。 


代码 位 置 ， 光盘 \TMNsI22\bcty365vadmin\savebccd.php 

<?php include_once("../conn/conn.php"); 

$bccdname=$_POST[bccdname]; 

S$owner=$_POST[owner]; 

$typeid=$_POST[typeid]; 

$content=$_POSTI[content]; 

$samepart=$_POST[samepart]; 

$addtime=date("Y-m-j Hii:s"); 

ifis_dir("./bccdimages")==false){ 

mkdir("./bccdimages"); 

上 

Slink=date("YmjHis"); 

// 为 表单 中 提交 的 数据 重新 命名 ， 以 当前 时 间 和 随机 数 作为 名 称 





// 连 接 数 据 库 
// 获 取 POST 方法 提交 的 值 


/获取 当前 时 间 
// 判 断 指定 的 文件 是 否 存在 
// 如 果 不 存在 ， 则 创建 一 个 新 的 文件 夹 


// 获 取 当 前 时 间 


// 其 中 使 用 $_FILES 获取 表单 中 真实 的 名 称 ， 使 用 strstr 函数 获取 文件 的 后 缀 
$path=$link.mt_rand(1000000,9999999).strstr($_FILES["imageaddress"]["name"],"."); 


$address="./bccdimages/".$path; 


// 定 义 文件 上 传 的 路 径 


move_uploaded_file($_FILES["imageaddress"]["tmp_name"],$address);// 将 文件 上 传 到 指定 的 文件 中 


Simageaddress="./admin/bccdimages/". $path; 
// 将 表单 中 提交 的 数据 存储 到 数据 库 中 


// 获 取 上 传 文件 在 服务 器 中 的 存储 路 径 


$query=mysql_query("insert into tb_bccd(bccdname,owner,typeid,content,samepart,imageaddress,addtime) 
values('$bccdname','$owner','$typeid','$content','$samepart','$imageaddress','$addtime'’)", $conn); 


if($query==true}{ 


echo "<script>alert(' 编 程 词典 添加 成 功 ! ');history.back();</script>"; 


jelsef 


echo "<script>alert( 编程 词典 添加 失败 ! ');history.back();</script>"; 


?> 


注意 


应 用 POST 方法 上 传 图 片 文件 时 ， 应 当 在 上 传 表单 的 <form> 标 记 中 添加 以 下 内 容 enctype= 


"multipartform-data"。 
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22.13.3 ”添加 编程 词典 的 实现 过 程 


添加 编程 词典 的 功能 是 向 数据 库 中 添加 编程 词典 的 详细 信息 ， 包 括 编程 词典 的 名 称 、 版 权 、 图 片 、 
类 别 、 内 容 简介 和 不 同 版 本 的 共同 点 。 其 运行 结果 如 图 22.47 所 示 。 
EL 六 


编程 词典 名 称 : PIF 篇 程 词典 
































版 权 所 有 : 。 | 车 条 省 明理 科技 有 有 限 信 司 
图 上 路 径 : | Ciocments ISRC EGG 
所 属 类 别 : 『 PHP 】 关 了 
| 
I 
内 容 简介 : 
Eee 习 
拉 届 不 同 的 碑 本 ,其中 涉及 到 不 同 台 思 程度 前 知 流 下 | 
不 同 版 本 相同 点 : 
加 
添加 | 重 写 








图 22.47 添加 编程 词典 模块 的 运行 结果 


添加 编程 词典 信息 模块 主要 通过 addbccd.php 和 savebccd.php 文件 来 完成 ， 其 中 在 addbccd.php 文 
件 中 主要 是 设计 表单 元 素 ， 而 savebccd.php 文件 主要 是 对 表单 中 提交 的 数据 进行 处 理 。addbccd.php 文 
件 中 使 用 的 表单 元 素 如 表 22.5 所 示 。 
表 22.5 添加 编程 词典 页 中 使 用 的 重要 表单 元 素 








名 称 | 元 素 类 型 重要 属性 含 义 
it Fa method="post" et "eayebeed Bhp onSubmit="retumn chkinput(this)" 编程 词典 表单 
enctype="multipart/form-data"> 
bccdname text Size="25" class="txt_grey" 编程 词典 名 称 
Owner text " size="25" class="txt 版 权 所 有 者 





imageaddress | file Size="25" class="txt grey" 界面 图 片 
<?php include once("../conn/conn.php"): 


$sql=mysql_query("select * from tb type order by createtime desc".$conn): 
S$info=mysql fetch array($sqD): 
i ee if($info 一 false){ echo "<option > 暂 无 类 别 </option>": 选择 编程 词典 
jelse{ 的 版 本 
do{ 


echo "<option value=".$info[id].">".$info[typename]."</option>": } 
While($info=mysql fetch array($sqD)):} ?> 
content textarea TOWS="10" cols="65" class="textarea"> 内 容 简介 
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续 表 
名 称 | 元 素 类 型 重要 属性 含义 
不 同 版 本 的 
samepart textarea IOWS="10" cols="65" class="textarea"> 特点 
Submit submit value=" 添 加 " class="btn grey" 提交 表单 
savebccd.php 文件 实现 对 表单 中 提交 的 数据 进行 处 理 ， 首 先 通过 $_POST 获取 表单 中 提交 的 数据 ， 


然后 判断 指定 的 文件 夹 是 否 存 在 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 其 关键 代码 如 下 。 


22. 


代码 位 置 ， 光盘 \TMN\sI\22\bcty365\admin\savebccd.php 

<?php include_once("../conn/conn.php"); // 连 接 数据 库 
$bccdname=$_POST[bccdname]; // 获 取 POST 方法 提交 的 值 
S$owner=$_POST[owner]; 

S$typeid=$_POST[typeid]; 

$content=$_POSTI[content]; 

$samepart=$_POST[samepart]; 


$addtime=date("Y-m-j H:i:s"); // 获 取 当 前 时 间 
ifis_dir("./bccdimages")==falsejf{ // 判 断 指定 的 文件 是 否 存 在 
mkdir("/bccdimages"); // 如 果 不 存在 ， 则 创建 一 个 新 的 文件 夹 
S$link=date("YmjHis"); // 获 取 当 前 时 间 


/为 表单 中 提交 的 数据 重新 命名 ， 以 当前 时 间 和 随机 数 作为 名 称 ， 其 中 使 用 $_FILES 获取 表单 中 真实 的 名 称 ， 使 用 
/strstr 函数 获取 文件 的 后 缀 
$path=$link.mt_rand(1000000,9999999).strstr($_FILES["imageaddress"]["name"],"."); 


$address="./bccdimages/".$path; /定义 文件 上 传 的 路 径 
move_uploaded_file($_FILES["imageaddress"]["tmp_name"],$address); // 将 文件 上 传 到 指定 的 文件 中 
$imageaddress="./admin/bccdimages/".$path; 1/ 获取 上 传 文件 在 服务 器 中 的 存储 路 径 

// 将 表单 中 提交 的 数据 存储 到 数据 库 中 


$query=mysql_query("insert into tb_bccd(bccdname,owner,typeid,content,samepart,imageaddress,addtime) 
values('$bccdname','$owner','$typeid','$content','$samepart','$imageaddress','$addtime’)", $conn); 
if($query==true){ 

echo "<script>alert(' 编 程 词典 添加 成 功 !");history.back();</script>"; 
Jelse{ 

echo "<script>alert(' 编 程 词典 添加 失败 !");history.back();</script>"; 
上 


?> 


13.4 ”编辑 编程 词典 的 实现 过 程 


在 完成 对 编程 词典 信息 的 添加 后 ， 接 下 来 就 可 以 对 编程 词典 的 版 本 信息 进行 编辑 ， 主 要 添加 版 本 





信息 、 价 格 、 简 介 、 功 能 和 推出 的 服务 。 该 模块 的 运行 结果 如 图 22.48 所 示 。 
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编辑 编程 词典 
编 程 词典 名 称 | 添加 时 间 添加 版 本 信息 删除 
1 编程 词典 | 200r-12-13 11:44:58 添加 全 
YP 编程 词典 | 2007-12-02 10:29:42 尖 加 钙 
版 本 名 称 内 容 简介 | 功能 服务 删除 
编程 间 典 名 称 : 四 
版 本 名 称 : 了 
价格 : 
加 
该 版 该 类 编程 词典 简 
介 : 
加 
琴 能 强大 司 
加 
服务 齐全 司 
ee 
加 
Ea 

















图 22.48 编辑 编程 词典 模块 的 运行 结果 
该 功能 的 实现 同样 通过 两 个 文件 ， 一 个 是 提交 表单 的 文件 editbccdphp， 另 一 个 是 处 理 表单 提交 数 
据 的 文件 sacvbccdbb.php。 提 交 表 单 文件 editbccd.php 中 使 用 的 表单 元 素 如 表 22.6 所 示 。 
表 22.6 编辑 编程 词典 页 中 使 用 的 重要 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含 义 
fe pe method="post" action="savebccdbb.php" onSubmit="return 编辑 编程 词典 表单 


chkinput(this)"> 
class="txt_grey" disabled="disabled" value=" 


<?php 
$sql4=mysql_query("select bccdname from tb_bccd where 编程 词典 名 称 ， 这 里 设 
| id=".$_GET[bccdid]."".$conn): 置 了 文本 的 只 读 属 性 


S$info4=mysql fetch_array($sql4); 
echo unhtml($info4[bccdname)]): ?> 
bccdid hidden value="<?php echo $_ GET[bccdid]:?> 编程 词典 的 ID 
<?php $sql3=mysql query("select * from tb_bb order by 
createtime desc ".$conn): 

S$info3=mysql fetch array($sql3):; 

这 $info3 一 包 lse){ echo “<option> 暂 无 版 本 信息 </option>"; 
bbid select }else{ 选择 编程 词典 的 版 本 

dof > 
<option value="<?php echo $info3[id]:?>"><?php echo 
unhtml($info3[bbname]):?></option> 


?php } while($info3=mysql fetch array($sql3)): }?> 


Submit submit Value=" 添 加 " class="btn grey” 提交 表单 














421 


MySQL 从 入 门 到 精通 


sacvbccdbb .php 文件 对 表单 提交 的 数据 进行 处 理 , 首先 获取 表单 中 提交 的 数据 , 然后 判断 指定 的 版 


本 是 否 已 经 被 添加 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 其 代码 如 下 。 


22. 


代码 位 置 ， 光 盘 \TM\sI\22\bcty365\admin\savebccdbb.php 


<?php 
$bccdid=$_POST[bccdid]; // 获 取 表 单 中 提交 的 数据 
$bbid=$_POST[bbid]; // 获 取 编 程 词 典 id 
$price=$_POSTI[price]; /获取 编程 词典 单价 
S$content=$_POST[content]; // 获 取 编 程 词典 内 容 
$gn=$_POST[gn]; // 获 取 编 程 词典 功能 
$fiw=$_POSTIfw]; /获取 编程 词典 服务 
include_once("../conn/conn.php"); /连接 数据 库 文件 
// 判 断 提交 的 编程 词典 是 否 已 经 被 添加 
$sql=mysql_query("select id from tb_bbqb where bccdid=".$bccdid."",$conn); 
$info=mysql_fetch_array($sql); /检索 指定 编程 词典 的 id 
if($info!=falseX{ // 如 果 检索 值 为 值 ， 则 弹出 提示 

echo "<script>alert(' 该 版 编程 词典 已 经 添加 !");history.back();</script>"; 

exit; } 
/人 
$query=mysql_query("insert into tb_bbqb(bccdid,bbid,price,content,gn,fw) 
values('$bccdid','$bbid','$price','$content','$gn','$fw')", $conn); // 将 表单 中 提交 的 数据 存储 到 数据 库 中 
// 更 新 编程 词典 的 价格 
$querys=mysql_query("update tb_bccd set bbid='$bbid',price='$price' where id=".$bccedid."™"); 
if($query==true and $querys==trueX{ /| 如果 添 加 和 更 新 操作 为 真 ， 则 弹出 提示 

echo "<script>alert(' 版 本 信息 添加 成 功 !");history.back();</script>"; 
Jelse{ // 如 果 添 加 和 更 新 操作 为 假 ， 则 弹出 提示 

echo "<script>alert(' 版 本 信息 添加 失败 !");history.back();</script>"; 
并 

A 大 ~ ~ 
22.14 软件 升级 管理 模块 设计 

14.1 软件 升级 管理 模块 概述 


软件 升级 管理 模块 实现 对 软件 升级 包 的 管理 ， 其 具体 的 功能 包括 添加 升级 包 、 编 辑 升级 包 、 添 加 





序列 号 和 编辑 序列 号 。 软 件 升 级 管理 模块 中 的 添加 升级 包 和 添加 序列 号 是 一 一 对 应 的 ， 其 中 根据 所 属 
的 类 别 和 版 本 来 确定 升级 包 对 应 的 序列 号 ， 每 一 个 版 本 一 个 类 别 的 升级 包 对 应 一 个 序列 号 。 


22. 


14.2 ”软件 升级 管理 模块 技术 分 析 


在 软件 升级 包 管理 模块 中 ， 应 用 到 一 个 动态 输出 下 拉 列 表 框 中 值 的 技术 。 下 面 就 来 讲解 一 下 该 技 








术 是 如 何 实现 的 ， 在 讲解 该 技术 之 前 ， 先 来 了 解 下 拉 列 表 框 的 基本 结构 。 
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<select name="select"><!->name 指定 该 下 拉 列 表 框 的 名 称 <!-> 
<L->selected 设置 下 拉 列 表 框 的 默认 值 ,默认 值 为 PHP<!--> 
<option selected="selected">PHP</option> 
<!-->value 指定 的 "mysql" 是 下 拉 列 表 框 传递 的 值 ,"'MYSQL" 为 显示 的 内 容 <!--> 
<option value="mysql">MYSQL</option> 
</select> 


所 谓 动态 输出 下 拉 列 表 框 中 的 值 就 是 从 数据 库 中 读 取 数 据 , 将 获取 到 的 数据 输出 到 下 拉 列 表 框 中 ， 
而 不 是 直接 在 下 拉 列 表 框 中 设置 某 个 固定 的 值 。 这 里 以 软件 升级 管理 模块 addsjb.php 文件 中 的 “所 属 
类 别 ” 下 拉 列 表 框 为 例 进行 讲解 ， 其 中 设置 下 拉 列 表 框 的 名 称 为 typeid， 默 认 值 为 “请 选择 ”，value 
的 值 是 从 数据 库 中 获取 的 ID 值 ， 显示 的 内 容 为 从 数据 库 中 获取 的 类 型 名 称 。 动 态 输出 下 拉 列 表 框 中 值 
使 用 的 关键 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\22\bcty365\ admin\wzdh.php 


< 上 设置 下 拉 列 表 框 的 名 称 为 ypeid 一 一 一 一 一 一 一 一 一 一 一 -> 
<select name="typeid" class="txt_grey"> 
<option value="" selected="selected"> 请 选择 </option> 











< 一 一 一 一 一 一 一 一 -一 一 一 一 设置 下 拉 列 表 框 的 名 称 为 ypeid 一 一 一 一 一 一 一 一 一 
<?php 

include_once("../conn/conn.php"); ll 连接 数据 库 

// 从 数据 库 中 读 取 编 程 词典 类 型 的 数据 


$sql=mysql_query("select * from tb_type order by createtime desc",$conn); 
$info=mysql_fetch_array($sql); 
if($info==false){ 

echo "<option > 暂 无 类 别 </option>"; 


jelsef 

dof /应 用 do…while 循环 语句 输出 类 型 的 ID 和 类 型 的 名 称 
echo "<option value=".$infofid].">".$info[typename]."</option>"; 

} 
while($info=mysql_fetch_array($sql)); /do…while 循环 语句 结束 

外 

?> 
</select> 


下 拉 列 表 框 不 但 可 以 动态 输出 数据 库 中 某 个 字段 的 数据 ， 而 且 可 以 输出 数组 中 的 数据 。 下 面 就 实 
现 一 个 在 下 拉 列 表 框 中 动态 输出 数组 中 数据 的 功能 ， 首 先 创建 一 个 下 拉 列 表 框 ， 然 后 设置 下 拉 列 表 框 
的 值 ， 从 数组 中 读 取 数据 ， 应 用 for 循环 语句 进行 输出 ， 其 代码 如 下 。 


和 一 一 一 一 一 一 一 设置 下 拉 列 表 框 的 名 称 为 select 一 一 一 一 一 一 一 一 一 一 一 一 > 

<select name="select" size="1"> 

<?php 
S$string="ASP,PHP,JSP,.NET,DEL,VB,VC": // 定 义 一 个 字符 串 
$srtings=split(",", $string); // 对 字符 串 进行 分 割 
$count=count($srtings); /获取 数 组 中 元 素 的 数量 
for($i=0:$i<$count:$i++)f // 根 据 数组 中 的 元 素 的 数量 进行 循环 输出 

Sresult=$srtings[$i]; // 定 义 变量 ， 获 取 数 组 中 指定 的 元 素 

echo "<option>S$result</option>";} /将 数组 中 的 元 素 输出 到 下 拉 列 表 框 中 
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se 
</select> 


动态 输出 数据 库 中 数据 到 下 拉 列 表 框 的 运行 结果 如 图 22.49 所 示 。 





编程 间 和 典 名 称 : PT 仿 程 司 奥 

















版 权 所 有 : 天 标明 日 村 有 限 公司 
图 片 路 径 : Eee wd Setine Re 
所 尾 基 别 : 请 选择 | 
过 拉 画 
内 容 简介 : 
习 











图 22.49 动态 输出 数据 库 中 数据 到 下 拉 列 表 框 


22.14.3 ”软件 升级 包 上 传 的 实现 过 程 


软件 升级 包 上 传 在 添加 升级 包 模块 中 实现 ， 通 过 一 个 文件 域 文 本 框 将 升级 包 提交 到 服务 器 中 指定 
的 文件 下 ， 并 且 将 该 文件 在 服务 器 中 的 路 径 存储 到 数据 库 中 ， 便 于 在 前 台 实 现 对 软件 升级 包 的 下 载 。 
其 运行 结果 如 图 22.50 所 示 。 


| 加 升级 包 
升级 包 名 称 : Pe 


| 
| -所属 类 别 : Tm “ 司 
| 
| 











版 本 名 称 : [标准 有 司 

















下 载 地 址 : C:\Documents and Settings\| 
FIP 物流 管理 系统 的 升级 包 = 
说 明 及 简介 : 
| 提交 | 重 写 





图 22.50 ”软件 升级 包 上 传 的 运行 结果 





在 本 模块 中 通过 addsjb.php 文件 来 提交 升级 包 的 信息 , 通过 savesj.php 文件 来 对 表单 提交 的 数据 进 





行 处 理 


E。 其 中 在 将 升级 包 上 传 到 服务 器 的 指定 文件 夹 的 过 程 中 ， 主 要 应 用 的 是 move_ uploaded file0 函 


数 。 在 savesjphp 文件 中 ， 首 先 获取 表单 提交 的 数据 ， 然 后 判断 服务 器 中 是 否 存在 指定 的 文件 ， 最 后 应 
用 move_uploaded_file0 函 数 将 升级 包 上 传 到 指定 的 文件 夹 下 ， 并 且 将 数据 存储 到 指定 的 数据 表 中 。 其 


程序 代码 如 下 。 
代码 位 置 : 光盘 \TMN\sl22\bcty365\admin\savesj.php 
<?php 
$name=$_POSTIname]; 1/ 获取 表单 提交 的 数据 
Stypeid=$_POSTI[typeid]; 1/ 获取 表单 提交 的 数据 
S$content=$_POST[content]; /获取 表单 提交 的 数据 


第 22 章 Apachet+PHP+MySQL 实现 网 上 社区 


$addtime=date("Y-m-j Hii:s"); // 定 义 时 间 变 量 
S$bbid=$_POST[bbid]; 1/ 获取 表单 提交 的 数据 
if(is_dir("./sjxz")==falseX{ / 淹 断 指定 的 文件 夹 是 否 存在 

mkdir("./sjxz"); // 如 果 指 定 的 文件 夹 不 存在 ， 则 创建 一 个 指定 的 文件 夹 
} 
Slink=date("YmjHis"); /获取 一 个 时 间 
$path=$link.mt_rand(1000000,9999999).strstr($_FILES["address"]["name","."); /重新 设置 升级 包 名 称 
$address="./sjxz/". $path; /设置 升级 包 在 服务 器 中 存储 的 指定 路 径 
move_uploaded_file($_FILES["address"]["tmp_name"],$address); // 将 升级 包 上 传 到 指定 的 路 径 下 
S$address="./admin/sjxz/".$path; // 获 取 升 级 包 在 服务 器 中 的 存储 路 径 
include_once("../conn/conn.php"); // 连 接 数 据 库 文件 


// 将 上 传 的 数据 存储 到 数据 库 中 ， 这 里 将 升级 包 在 服务 器 中 的 路 径 存储 到 数据 库 中 
$query=mysql_query("insert into tb_sjxz(name,typeid,content,addtime,address,bbid) 
values('$name','$typeid','$content','$addtime','$address','$bbid')", $conn); 


if($query\{ /如果 添 加 操作 成 功 ， 则 弹出 提示 
echo "<script>alert(' 升 级 包 添加 成 功 !"");history.back();</script>"; 

Jelse{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 
echo "<script>alert( 升 级 包 添加 失败 !");history.back();</script>"; 

} 

ws 


22.14.4 ”软件 升级 包 删 除 的 实现 过 程 


软件 升级 包 删 除 的 实现 主要 根据 当前 数据 中 提供 的 ID， 执行 delete 删除 语句 ， 将 数据 表 中 相同 ID 
的 数据 删除 。 其 运行 结果 如 图 22.51 所 示 。 


四 你 当前 的 位 置 : 编 和 久生 包 














田 技术 支持 管理 i 
| 有 纺 久 和约 包 

时 软件 试用 管理 _ 升级 包 版 本 | ”升级 天 别 升 当 包 名 冤 Na | 。 蝇 辽 

区 人 下 全 | 。 商务 乒 VB 抱 生 词 内 3 0 天 红包 2007-12-02 14:32:51 | 个 | 
共有 升 红外 1 个 全 1 





喜 户 服务 中 心 热 绩 : 0431-64978961 Copyri 和 ht 2007 四) 吉林 省 明日 科技 有 限 公司 
图 22.51 软件 升级 包 删除 的 运行 结果 
该 功能 的 实现 主要 通过 editsjb.php 文件 和 deletesjb php 文件 。 通 过 editsjb.php 文件 输出 数据 库 中 存 
储 的 有 关 升 级 包 的 信息 ， 以 分 页 的 形式 显示 ， 在 每 条 记录 的 最 后 设置 一 个 删除 链接 ， 通 过 脚本 来 调 
deletesjb.php 文件 ， 根 据 变量 中 的 ID 值 执行 删除 升级 包 的 操作 。 其 关键 代码 如 下 。 

















425 


MySQL 从 入 门 到 精通 


代码 位 置 : 光盘 \TMNsI22Vbcty36svadminvdeletesjbphp 


<?php 
$id=$_GET[id]; /获取 变量 传递 的 ID 
include_once("../conn/conn.php"); // 连 接 数据 库 


/执行 删除 操作 ， 将 数据 表 中 对 应 的 ID 的 数据 删除 
if(mysql_query("delete from tb_sjxz where id=".$id."",$conn))f 


echo "<script>alert(' 该 升级 包 删 除 成 功 !");history.back();</script>"; // 如 果 删 除 操作 成 功 , 则 弹出 提示 
Jelse{ // 如 果 删 除 操作 失败 , 则 弹出 提示 
echo "<script>alert(' 该 升级 包 删 除 失败 "");history.back();</script>"; 
} 
2 


22.15 在 Linux 系统 下 发 布 网 站 


Linux 下 发 布 基 于 PHP 的 网 站 ， 首 先 需 要 配置 PHP 的 运行 环境 ， 其 次 需要 对 网 卡 参 数 进 行 设 定 。 
这 里 将 以 发 布 “BCTY365 网 上 社区 ”网 站 为 例 讲解 Linux 下 如 何 实 现 网 站 的 发 布 。 假 设 已 经 申请 到 如 
表 22.7 所 示 的 网 络 参数 。 


表 22.7 ”申请 到 的 网 络 参数 


参数 值 

了 192.168.1.* 
Netmask 255.255.255.* 
Network 192.168.1.0 
Broadcast 192.168.1.* 
Gatewa 192.168.1.* 
主机 名 Tsoft 

DNS 168.95.1* 


Linux 下 网 站 发 布 的 操作 步 又 如 下 。 
(1) 配置 PHP 的 运行 环境 ， 在 22.4 节 中 已 经 做 了 详细 介绍 ， 这 里 不 再 袭 述 。 
(2) 将 “BCTY365 网 上 社区 ”网 站 的 所 有 文件 复制 到 Apache 主 目录 下 。 
(3) 设置 主机 名 称 。 在 终端 窗口 中 输入 如 下 命令 编辑 /etc/sysconfig/network 文件 。 
Vi /etc/sysconfig/network 
将 该 文件 中 的 参数 NETWORKING 设置 为 yes, 表示 启动 网 络 , 将 参数 HOSTNAME 设置 为 Tsoft， 
表示 设置 主机 名 为 Tsoft。 
(4) 设置 网 卡 参数 。 在 终端 窗口 中 输入 如 下 命令 编辑 文件 /etc/sysconfig/network-scripts/ifcfg-eth0。 
vi /etc/sysconfig/network-scripts/ifcfg-ethO 


该 文件 的 相关 参数 设置 如 表 22.8 所 示 。 
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表 22.8 设置 网 卡 的 相关 参数 























参数 说 明 
DEVICE=eth0 设置 网 卡 名 称 ， 要 与 这 人 -eth0 对 应 
ONBOOT=yes 指定 在 开机 时 启动 网 卡 
BOOTPROTO=static 设 定 启动 时 获取 下 的 方式 
IPADDR=192.168.1.* 设 定 服务 器 他 地 址 
NETMASK=255.255.255.* 设 定子 网 掩 码 
BROADCAST=192.168.1.* 设 定 同 网 段 的 广播 地 址 
GETWAY=192.168.1.* 设 定 网 卡 的 网 关 


(5) 设置 DNS 主机 的 卫 。 在 终端 编辑 /etc/resolv.conf 文件 : 
vi /etc/resolv.conf 
设置 参数 nameserver 的 值 为 168.95.1.*。 

(6) 重新 启动 网 络 设置 。 在 终端 窗口 中 输入 如 下 命令 。 


letc/re.d/inin.d/network restart 
ifdow eth0 
ifup eth0 
(7) 打开 浏览 器 , 在 地 址 栏 中 输入 服务 器 卫 地 址 或 域名 , 打开 如 图 22.52 所 示 的 页 面 , 说 明 Linux 
下 “BCTY365 网 上 社区 ”网 站 发 布 成 功 。 











文 作 (E) 坊 辑 (E) 查看 (V) 转 到 (G)》 书签 (B) 工具 ID 窗口 WD 帮助 0H) 





总 - 宫 - 访 、 革 [http//192 168.1.59/TsoftUbety365/ 口 区 是 此 - 


从 主页 | 由 书签 和 Red Hat Network 茵 Support 冰 Shop 贡 Prodocts 前 Tenng 


















在 线 订购 升级 下 载 
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泥 汪 这 | 完成 | -| 


Ve SS Te) | | "由 


图 22.52 Linux 下 “BCTY365 网 上 社区 ”网 站 运行 结果 
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22.16 ”开发 技巧 与 难点 分 析 


22.16.1 ”管理 员 权 限 的 设置 


为 了 更 好 地 管理 和 维护 网 站 的 论坛 ， 针 对 论坛 设置 了 一 个 管理 员 ， 该 管理 
而 是 在 前 台 为 管理 员 设 置 特殊 的 权限 ， 也 可 以 称 之 为 版 主 。 其 实现 的 原理 是 : 


员 不 在 后 台 进 行 操作 ， 
首先 在 数据 库 中 设置 不 


同 的 值 代表 不 同 的 权限 ，“0” 代 表 普 通 会 员 ，“1” 代 表 管 理 员 ;然后 在 论坛 的 页 面 中 进行 判断 ， 当 


用 户 的 类 型 为 “1” 时 ， 不 但 具有 普通 会 员 的 权限 ， 而 且 具 有 删除 发 布 帖子 、 回 





如 果 用 户 的 类 型 不 是 “1”， 则 不 具有 上 述 的 权限 ， 只 能 是 发 布 和 回复 帖子 。 管 
页 面 效 果 是 不 同 的 ， 如 图 22.53 和 图 22.54 所 示 。 





起 请 移 一 下 如 何 职 置 nz 的 环境 


EEC mm 


2001-t9-12 13:21 但 ED 
是 境 和 





MA 
hess 


用 记名 别 : 管理 员 
区: 
丝 玉 时 间 : 200T-1z-0L 13 15 25 


图 22.53 管理 员 登 录 的 操作 页 面 


[ER 





复 帖 子 和 项 帖 的 权限 ; 
理 员 和 普通 会 员 登录 的 
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发 贴 人 : TSoft 
信箱 国 oica 旺 IPt 址 
用 户 全 别 : 






注册 时 间 : 2007-11-18 08:42:43 


ED CD 


图 22.54 普通 会 员 登 录 的 操作 页 面 
在 页 面 中 执行 的 判断 语句 判断 登录 用 户 的 类 型 , 然后 根据 类 型 判断 用 户 的 权限 


<?php 


。 其 程序 关键 代码 如 下 。 


if($_SESSION["unc"]!=™"X{ /判断 session 变量 的 值 是 否 为 空 


/根据 session 变量 的 值 获取 该 用 户 的 类 型 


$sqlu=mysql_query("select usertype from tb_user where usernc=".$_SESSION["unc"]."",$conn); 
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$infou=mysql_fetch_array($sqlu); 
if($infou["usertype"]==1X{ 1/ 判断 当 该 用 户 的 类 型 等 于 1 时 执行 下 面 的 内 容 


Ee 

<!--> 调 用 JavaScript 脚本 ， 执 行 删除 发 布 帖子 的 操作 <!--> 

<img src="images/lt_15(10).jpg" onclick="javascript:if(window.confirm(' 您 确定 删除 该 帖 么 ? 
'")==true}{window.location.href='bbs_delete.php?id=<?php echo S$infob["id"]?>";}" style="cursor:hand"/> 
<?php 


} 
1/ 如果 用 户 的 类 型 不 是 1， 则 不 执行 上 述 的 内 容 


22.16.2 ”帖子 置顶 的 设置 


所 谓 帖子 置顶 就 是 将 某 个 指定 的 帖子 在 对 应 的 版 块 中 最 前 面 的 位 置 显示 ， 该 权限 只 有 管理 员 才 拥 
有 ， 普 通 会 员 不 具备 该 权限 。 其 实现 的 原理 如 下 。 

首先 ， 在 数据 库 中 存储 发 布 帖子 信息 的 数据 表 中 设置 一 个 字段 top， 指 定 该 字段 为 数字 类 型 ， 其 默 
认 值 为 0。 

然后 ， 在 网 页 中 判断 登录 用 户 的 权限 ， 如 果 是 管理 员 ， 则 具有 帖子 置顶 的 权限 ， 否 则 将 弹出 提示 
对 话 框 “对 不 起 ， 您 不 具备 该 操作 权限 ! ”。 

最 后 ， 如 果 是 管理 员 ， 则 执行 settop.php 文件 ， 根 据 对 应 帖子 的 ID 查找 到 发 布 帖子 信息 表 中 对 应 
的 数据 ， 更 新 该 条 数据 中 top 字段 的 值 为 1。 

判断 登录 用 户 权 限 使 用 的 代码 如 下 。 


<?php 

if($_SESSION["unc"]=="™, // 判 断 session 变量 的 值 是否 为 空 ， 如 果 为 空 则 执行 下 面 的 脚本 
echo "javascript:alert(' 请 先 登录 本 站 ， 然 后 进行 此 操作 !");window.location.href="index.php';"; 

Jelse{ 
// 如 果 不 为 空 则 执行 下 面 的 内 容 ， 判 断 登录 用 户 的 权限 
$sqlu=mysql_query("select usertype from tb_user where usernc=".$_SESSION["unc"]."",$conn); 
S$infou=mysql_fetch_array($sqlu); 
if($infou["usertype"]==1X{ // 如 果 登 录用 户 的 类 型 是 "1"， 则 说 明 是 管理 员 ， 则 执行 下 面 的 内 容 

echo "javascript:window.location.href='settop.php?id=".$infob["id"].""; 





jelsef 
// 如 果 登 录用 户 不 是 管理 员 ， 则 执行 下 面 的 内 容 
echo "javascript'alert(' 对 不 起 ， 您 不 具备 该 操作 权限 由 :"; 


} 
?> 
实现 帖子 置顶 是 通过 settop php 文件 来 完成 的 , 在 该 文件 中 , 根据 变量 提交 的 值 获取 到 发 布 帖子 信 
息 表 中 对 应 的 数据 ， 更 新 该 条 数据 中 字段 top 的 值 ， 并 且 对 该 字段 的 值 进行 判断 。 如 果 字 段 top 的 值 为 
1， 则 说 明 该 帖 已 经 置顶 ， 此 时 将 字段 的 值 更 新 为 0， 即 取消 置 项， 如 果 字 段 top 的 值 为 0， 则 说 明 该 
帖 没 有 进行 置顶 ， 此 时 将 字段 的 值 更 新 为 1， 即 置顶 该 帖 。settop.php 文件 的 程序 代码 如 下 。 
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<?php include_once("conn/conn.php"); /连接 数据 库 文件 
/根据 获取 的 ID 值 ， 从 数据 表 中 读 取 到 对 应 的 数据 

$sql=mysql_query("select top from tb_bbs where id=".$_GET[T"id"]."",$conn); 

$info=mysql_fetch_array($sql); 

if($info[ltop]==1X{ /判断 对 应 数据 记录 中 的 字段 top 的 值 ， 如 果 字 段 top 的 值 为 "1"， 则 执行 下 面 的 内 容 


mysql_query("update tb_bbs set top=0 where id=".$_GET["id"]."",$conn); // 更 新 字段 top 的 值 为 0 
}elseif($info[top]==0X{ // 如 果 对 应 数据 记录 中 的 字段 top 的 值 为 "0"， 则 执行 下 面 的 内 容 
mysql_query("update tb_bbs set top=1 where id=".$_GET["id"]."",$conn); // 更 新 字段 top 的 值 为 1 


' 
echo "<script>alert(' 置 上 设置 成 功 !"");history.back();</script>"; 
?> 


22.16.3 ”解决 提示 “客户 反馈 发 表 失 败 ! ”的 问题 


在 对 客户 反馈 模块 进行 测试 的 过 程 中 ， 当 提交 反馈 信息 时 ， 提 示 “ 客 户 反 馈 发 表 失 败 ! ”。 经 过 
初步 的 判断 ， 问 题 可 能 出 现在 向 数据 库 添加 数据 的 过 程 中 。 为 进一步 查找 出 问题 的 根源 ， 在 
saveleaveword.php 页 中 应 用 了 mysql_erorO 函 数 ， 该 函数 可 获取 详细 的 错误 信息 ， 将 它 放 置 在 insert 添 
加 语句 之 后 ， 然 后 重新 运行 网 页 ， 运 行 结果 如 图 22.55 所 示 。 





//102 168. 1. 59/er /bety305 /ere le 























在 该 结果 图 中 可 以 看 出 是 数据 表 中 的 type 字段 出 了 问题 , 经 过 仔细 检查 发 现 , 原来 是 在 编写 insert 
添加 语句 的 过 程 中 ， 将 语句 中 字段 的 排列 顺序 与 对 应 的 字段 值 的 顺序 弄 错 了 ， 如 图 22.56 所 示 。 










echo "<script>alert ("留言 发 表 成 功 ! *) ;hi 
}else{ 
echo“《script>alert( 留言 发 表 失 败 !“) :history.back() </script>"; 





颠倒 $title 和 $type 的 顺序 





图 22.56 ”代码 中 的 错误 截图 


调整 变量 $title 和 $type 的 顺序 ， 重 新 运行 程序 ， 提 示 “ 留 言 添加 成 功 ”。 





22.16.4 ”解决 指定 商品 没有 被 删除 的 问题 
在 在 线 订购 模块 的 测试 过 程 中 ， 当 执行 删除 购物 车 中 某 个 商品 的 操作 时 ， 发 现 该 项 操作 没 能 正确 
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执行 ， 指 定 的 商品 没有 被 删除 。 分 析出 错 原 因 : 首先 ， 从 该 功能 实现 的 思路 入 手 ， 并 且 仔 细 核 对 其 中 
使 用 的 变量 以 及 函数 是 否 正确 。 

删除 商品 的 操作 是 根据 商品 的 ID 来 进行 处 理 的 ， 当 单 击 “ 删 除 ” 超 链接 时 ， 将 指定 商品 的 ID 通 
过 变量 $ides 传 入 到 执行 删除 商品 操作 的 文件 delgoods.php 中 ， 然 后 在 delgoods.php 文件 中 ， 获 取 变 量 
$ides 传 入 的 ID 值 ， 并 且 根 据 ID 值 完成 删除 购物 车 中 指定 商品 的 操作 。 在 查看 该 功能 实现 的 过 程 中 发 
现 ， 定义 的 变量 $ides 与 delgoods.php 文件 中 $_GET["id"] 获 取 的 变量 值 不 一 致 ， 从 而 导致 删除 商品 的 操 
作 没有 正确 执行 ， 代 码 的 出 错位 置 如 图 22.57 所 示 。 

shopping_cart. php 文 件 中 执行 删除 操作 的 内 容 









td height="22" bgcolor="#FFFFFF” 
《div alignr conter” > 
a href= javascript;jFarak2nhn echo a el submit () ;”class="al" > 更 改 数量 人 /a 
《<a href="q ?ide 六 d] ;3?>”class="al“ 7 删除 该 项 </a> 
/div> 





定义 与 获取 的 变量 值 不 一 致 定义 传递 ID 值 的 变量 
delgoods. php 文 件 中 执行 删除 操作 的 内 容 


tart () ; 









Pe 人 N["goodsid“]) 
$arraynun=explode (“@”, JN[” goodsnum”]) ; 

Soyarrey soarch (sid ayid) ; 

$arrayid[$key]="" 

$arraynun [$key]="" 

$. “goodsid’ ]=inplode ("@", $arrayid) : 

“goodsnun” ]=inplode (“@, $arraynun) : 

ipt>window. location href=" shopping_cart,php ;</script》 









图 22.57 ”代码 中 的 错误 显示 


查找 出 错误 原因 ， 将 定义 的 变量 与 获取 的 变量 值 统一 使 用 id， 保 存 文件 ， 重 新 运行 程序 ， 删 除 操 
作 可 以 正常 运行 。 


22.16.5 ”解决 发 帖 和 回帖 时 上 传 的 图 片 不 能 够 正常 显示 的 问题 


在 测试 网 上 社区 的 论坛 模块 时 , 发 现 发 帖 和 回帖 时 上 传 的 图 片 不 能 够 正常 显示 。 运行 结果 如 图 22.58 
所 示 。 分 析 错 误 原 因 : 主要 有 两 个 方面 ， 一 是 图 片 没有 上 传 到 指定 的 服务 器 文件 夹 下 ; 二 是 上 传 成 功 
后 ， 没 能 正确 地 读 取 数据 库 中 指定 图 片 的 路 径 。 

因 当 首位 置 >》 社区 论坛 论坛 版 六 ” 二 著 帖 子 局 














图 22.58 程序 运行 错误 结果 
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首先 ， 检 测 第 一 种 情况 ， 没 有 出 现 问题 ， 图 片 可 以 上 传 到 指定 的 文件 夹 下 ， 并 且 图 片 的 路 径 也 可 
以 存储 到 指定 的 数据 表 中 ， 从 而 排除 了 第 一 种 情况 的 可 能 。 

然后 ， 看 第 二 种 情况 ， 查 看 获取 的 图 片 路 径 是 否 正确 。 发 现在 读 取 数据 库 中 图 片 路 径 的 代码 段 中， 
使 用 了 错误 的 字段 名 称 ， 数据库 中 图 片 路 径 存 储 使 用 的 字段 名 是 photo， 而 在 程序 代码 段 中 使 用 的 却 是 
photos。 错 误 代码 如 下 。 


<?php 
if($infob[photo]!="™"){ /| 判断 是 否 存在 图 片 
$photos=substr($infob[photos],2,70); /获取 图 片 存储 的 路 径 
echo (stripslashes($infob["content"])); /| 输出 图 片 的 内 容 
echo "<img src=\"$photos\">"; /| 输出 图 片 
Jelse{ 
echo (stripslashes($infob["content"])); /| 输出 帖子 内 容 
} 
?> 


将 代码 段 中 的 字段 名 进行 修改 ， 然 后 重新 运行 程序 ， 图 片 正常 显示 。 
22.17 站 结 


本 章 从 项 目 开 发 的 实际 角度 出 发 ， 以 某 科 技 公司 的 实际 需求 为 背景 ， 详 细 地 讲解 了 “BCTY365 网 
上 社区 ”系统 的 开发 过 程 ， 其 中 以 系统 的 整体 开发 流程 为 主线 ， 重 点 介绍 技术 支持 、 在 线 订购 、 社 区 
论坛 和 编程 词典 等 几 个 大 模块 的 实现 方法 ， 并 且 对 管理 员 权限 设置 、 帖 子 置顶 设置 进行 讲解 ， 而 且 在 
本 章 中 还 讲解 了 在 Linux 下 如 何 搭建 PHP 的 开发 环境 以 及 在 Linux 下 如 何 发 布 网 站 。 
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喜欢 网 上 购物 的 读者 一 定 登 录 过 淘宝 ， 也 一 定 被 网 页 上 琳琅 满目 的 商品 所 吸 
3 引 ， 忍 不 住 拍 一 个 自己 喜爱 的 商品 。 如 今 也 有 越 来 越 多 的 人 加 入 到 网 购 的 行列 ， 做 
网 上 店铺 的 老板 , 做 新 时 代 的 购物 潮 人 ， 你 是 否 也 想 过 开发 一 个 自己 的 网 上 商城 ? 
本 章 我 们 将 一 起 进入 GO 购 网 络 商城 开发 的 旅程 。 

通过 阅读 本 章 ， 读 者 可 以 : 


Wp ”了解 网 上 商城 的 核心 业务 
MH 党 握 网 站 开发 的 基本 流程 


MySQL 从 入 门 到 精通 


23.1 开 发 背 景 


近年 来 ， 随 着 Intemet 的 迅速 崛起 ， 互 联网 用 户 的 爆炸 式 增长 以 及 互联 网 对 传统 行业 的 冲击 让 其 
成 为 人 们 快速 获取 、 发 布 和 传递 信息 的 重要 渠道 。 于 是 电子 商务 逐渐 流行 起 来 ， 越 来 越 多 的 商家 在 
网 上 建 起 网 上 商城 ， 向 消费 者 展示 出 一 种 全 新 的 购物 理念 ， 同 时 也 有 越 来 越 多 的 网 友 加 入 到 了 网 上 
购物 的 行列 。 阿 里 巴巴 旗下 的 淘宝 的 成 功 展现 了 电子 商务 网 站 强大 的 生命 力 和 电子 商务 网 站 更 加 光 
明 的 未 来 。 

为 了 利用 Internet 这 个 平台 , 实现 一 种 全 新 的 购物 方式 一 一 网 上 购物 , 其 目的 是 方便 广大 网 友 购 物 ， 
让 网 友 足 不 出 户 就 可 以 得 商城 买 商品 ， 为 此 构建 Go 购 商城 系统 。 


23.2 系统 分 析 


23.2.1 需求 分 析 


Go 购 商城 系统 是 基于 B/S 模式 的 电子 商务 网 站 ， 用 于 满足 不 同人 群 的 购物 需求 ， 笔 者 通过 对 现 有 
的 商务 网 站 的 考察 和 研究 ， 从 经 营 者 和 消费 者 的 角度 出 发 ， 以 高 效 管理 、 满 足 消费 者 需求 为 原则 ， 要 
求 本 系统 满足 以 下 要 求 。 

(1) 统一 友好 的 操作 界面 ， 具 有 良好 的 用 户 体验 ; 

(2) 商品 分 类 详尽 ， 可 按 不 同类 别 查 看 商品 信息 ; 

(3) 推荐 产品 、 人 气 商品 以 及 热 销 产品 的 展示 ; 

(4) 会 员 信息 的 注册 及 验证 ; 

(5) 用 户 可 通过 关键 字 搜 索 指定 的 产品 信息 ; 

(6) 用 户 可 通过 购物 车 一 次 购买 多 件 商品 ; 

(7) 实现 收银 台 的 功能 ， 用 户 选择 商品 后 可 以 在 线 提交 订单 ; 

(8) 提供 简单 的 安全 模型 ， 用 户 必须 先 登录 ， 才 人 允许 购买 商品 ; 

(9) 用 户 可 查看 自己 的 订单 信息 ; 

(10) 设计 网 站 后 台 ， 管 理 网 站 的 各 项 基本 数据 ; 

(11) 系统 运行 安全 稳定 、 响 应 及 时 。 








23.2.2 可行 性 分 析 
在 正式 开发 系统 之 前 ， 首 先 需要 对 Go 购 商城 的 技术 、 操 作 和 经 济 成 本 3 个 方面 进行 可 行 性 
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分 析 。 
1. 技术 可 行 性 


本 系统 采用 MVC 设计 模式 ， 使 用 当前 最 流行 的 Struts 2+Spring 2+Hibernate 框架 进行 开发 ， 在 前 
台 用 JSP 进行 页 面 开 发 和 管理 用 户 界面 ， 利 用 轻巧 的 JavaScript 库 一 一 jQuery 处 理 页 面 的 JavaScript 脚 
本 ， 使 开发 更 加 高 效 ， 提 示 信 息 更 加 完善 ， 界 面 友好 ， 具 有 较 强 的 亲和力 。 后 台 采 用 MySQL 数据 库 ， 
MySQL 小 巧 高 效 的 特点 可 以 满足 系统 的 性 能 要 求 。 本 系统 使 用 当前 主流 的 Java 开源 开发 工具 Eclipse 
和 Tomcat 服务 器 进行 程序 开发 和 发 布 ， 它 们 是 完全 免费 的 ， 可 以 节约 开发 成 本 。 本 系统 采用 的 技术 和 
开发 环境 在 实际 开发 中 应 用 非常 广泛 ， 充 分 说 明 本 系统 在 技术 方面 可 行 。 

2. 操作 可 行 性 

Go 购 商 城主 要 面向 的 是 喜欢 网 购 的 网 友 ， 只 要 Go 购 商城 的 用 户 会 一 些 简 单 的 计算 机 操作 ， 就 可 
以 进行 网 上 购物 ， 不 需要 用 户 具有 较 高 的 计算 机 专业 知识 ， 而 且 对 于 网 站 基本 信息 的 维护 也 是 十 分 简 
单 ,管理 员 可 以 在 任何 一 台 可 以 上 网 的 机 器 上 对 网 站 进行 维护 , 网 站 的 简单 易 用 性 充分 说 明了 Go 购 商 
城 的 操作 可 行 性 。 


3. 经 济 成 本 可 行 性 


在 实际 的 销售 运营 过 程 中 ， 产 品 的 宣传 受到 限制 ， 采 购 商 或 顾客 只 能 通过 上 门 咨询 、 电 话 沟通 等 
方式 进行 各 种 产品 信息 的 获取 ， 而 且 时 间 与 物理 的 局 限 性 严重 影响 了 产品 的 销售 ， 并 且 在 无 形 中 提高 
了 产品 的 销售 成 本 。 本 系统 完全 可 以 改变 这 种 现状 ， 以 少量 的 时 间 和 资金 建立 企业 商务 网 络 ， 以 此 来 
使 企业 与 消费 者 之 间 的 经 济 活动 变 得 更 加 灵活 、 主 动 。 

系统 中 应 用 的 开发 工具 以 及 技术 框架 都 是 免费 的 ， 这 无 疑 为 网 站 的 成 本 再 一 次 压缩 了 空间 ， 从 成 
本 可 行 性 分 析 来 看 ， 该 系统 充分 体现 了 将 产品 利益 最 大 化 的 企业 原则 。 





23.3 系统 设计 


23.3.1 功能 结构 图 


Go 购 商 城 系统 分 为 前 台 和 后 台 两 个 部 分 的 操作 。 前 台 主 要 有 两 大 功能 ， 分 别 是 展示 产品 信息 的 各 
种 浏览 操作 和 会 员 用 户 购买 商品 的 操作 ， 当 会 员 成 功 登 录 后 ， 就 可 以 使 用 购物 车 进行 网 上 购物 。Go 购 
商城 前 台 功 能 结构 图 如 图 23.1 所 示 。 
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GO 购 网 络 商城 
前 台 管 理 





a 商品 浏览 
吓 热 | 推 | 区 
全 | 全 | 多 | | 闫 | 全 | 菏 | 准 | 划 
员 | 员 | 霹 | 单 | 六 | 高 | 产 | 产 | 衬 
注 | | 痘 1 | 典 | | 坦 | | 六] 品 | | 品 | | 品 | | 的 
证 | 录 | 营 | 询 | 洛 | 河 | 沐 | 汶 | 案 
品 品 | | 将 | 奖 | 奖 | 品 
蝇 





图 23.1 Go 购 商 城 系 统 前 台 功 能 结构 图 


后 台 的 主要 功能 是 当 管 理 员 成 功 登录 后 台 后 ， 用 户 可 以 对 网 站 的 基本 信息 进行 维护 。 例 如 ， 管 
理 员 可 以 对 商品 的 类 别 进 行 管理 ， 如 可 以 删除 和 添加 产品 的 类 别 ， 可 以 对 商品 信息 进行 维护 ， 可 以 
添加 、 删 除 、 修 改 和 查询 产品 信息 ， 并 上 传 产品 的 相关 图 片 ， 可 以 对 会 员 的 订单 进行 集中 管理 ， 管 理 
员 可 以 对 订单 信息 进行 自 定 义 的 条 件 查询 并 修改 指定 的 产品 信息 。Go 购 商 城 后 台 功 能 结构 图 如 图 23.2 
所 示 。 
GO 购 网 络 商城 
后 台 管理 





类 别管 理 产品 管理 订单 管理 


让 

滩 苹 涯 泣 
让 
人 


器 穴 淮 劳 攻 瑟 
Et 
熙 陶 芭 三 冯 所 瑟 





23.2 ”Go 购 商城 系统 后 台 功能 结构 图 
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23.3.2 ”系统 流程 图 


在 Go 购 商 城中 只 有 会 员 才 允许 进行 购物 操作 , 所 以 初次 登录 网 站 的 用 户 如 果 想 进行 购物 操作 必 
须 注册 为 Go 购 商 城 的 会 员 。 成 功 注册 为 会 员 后 ,会 员 可 以 使 用 购物 车 选择 自己 需要 的 商品 ， 在 确认 
订单 付款 后 ， 系 统 将 自动 生成 此 次 交易 的 订单 基本 信息 。 网 站 基本 信息 的 维护 由 网 站 管理 员 负 责 ， 
由 管理 员 负 责 对 商品 信息 、 商 品类 别 信息 以 及 订单 信息 进行 维护 ， 关 于 订单 的 维护 只 能 修改 订单 的 
状态 ， 并 不 能 修改 订单 的 基本 信息 ， 因 为 订单 确认 之 后 就 是 用 户 与 商家 之 间 交 易 的 凭证 ， 第 三 方 无 
权 修 改 。 

Go 购 商 城 的 系统 流程 图 如 图 23.3 所 示 。 


























23.3 ”Go 购 商城 系统 流程 图 


23.3.3 ”开发 环境 


在 进行 Go 购 商 城 网 站 开发 时 ， 需 要 具备 以 下 开发 环境 。 

1. 服 务 器 端 

操作 系统 : Windows 2003 或 者 更 高 版 本 的 服务 器 操作 系统 。 
Web 服务 器 : Tomcat 7.0 或 7.0 以 上 版 本 。 

Java 开发 包 : JDK 1.7 以 上 。 

数据 库 : MySQL。 





MySQL 从 入 门 到 精通 
浏览 器 : 正 6.0 或 者 更 高 版 本 的 浏览 器 。 
分 辨 率 : 最 低 要 求 为 1024X768 像素 。 
2. 客 户 端 
浏览 器 : IE 6.0 或 者 更 高 版 本 的 浏览 器 。 
分 辩 率 : 最 低 要 求 为 1024X768 像素 。 
23.3.4 文件 夹 组 织 结构 


在 编写 代码 之 前 ， 可 以 把 系统 中 可 能 用 到 的 文件 夹 先 创建 出 来 〈 例 如 ， 创 建 一 个 名 为 images 的 文 
件 夹 ， 用 于 保存 网 站 中 所 使 用 的 图 片 》， 这 样 不 但 可 以 方便 以 后 的 开发 工作 ， 也 可 以 规范 网 站 的 整体 
架构 。 本 系统 的 文件 夹 组 织 结构 如 图 23.4 所 示 。 





4 上 Shop 
hh src 用 于 保存 Java 文 件 
4 hb WebContent 
< 和 cs Co Css 文件 
入 images 一 一 一 一 一 一 一 一 保 导 网 站 的 图 片 
局 Database 一 一 一 一 一 一 一 一 保存 数据 并 文件 
bj 一 化 Flavascript 文 件 
上 META-INF 
各 upload 一 保存 上 传 图 片 文件 
hh WEB-INF 
bk lib 一 人 保 )ar 包 文件 
和 pages 一 一 一 一 一 一 一 一 一 全 网 站 /SP 文件 








图 23.4 Go 购 商 城 文件 夹 组 织 结构 
23.3.5 ”系统 预览 


系统 预览 将 以 用 户 交易 为 例 ， 列 出 几 个 关键 的 页 面 ， 商 品 交易 是 Go 购 商 城 的 核心 模块 之 一 , 通过 
该 预览 的 展示 , 读者 可 以 对 Go 购 商 城 有 个 基本 的 了 解 , 同时 读者 也 可 以 在 光盘 中 对 本 程序 的 源 程序 进 
行 查看 。 

当 用 户 在 地 址 栏 中 输入 Go 购 商 城 的 域名 ， 就 可 以 进入 Go 购 商 城 ， 首 页 将 商品 的 类 别 信息 分 类 展 
现 给 用 户 ,并 在 首页 展示 部 分 的 人 气 商品 、 推 荐 商品 、 热 销 商品 以 及 上 市 新 品 ， 首 页 部 分 效果 如 图 23.5 
所 示 。 
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图 23.5 Go 购 商 城 首 页 部 分 页 面 效果 图 
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果 用 户 为 会 员 登 录 后 就 可 以 直接 进行 产品 的 选 购 ， 当 用 户 在 商品 信息 详细 页 面 中 单 击 “ 直 接 购 


买 ” 超 链 接 ， 就 会 将 该 商品 放 入 购物 车 中 ， 同 时 用 户 也 可 以 使 用 购物 车 选 购 多 种 商品 ， 购 物 车 同时 可 


以 保存 多 件 会 员 采 购 的 商品 信息 。 图 23.6 为 用 户 选 购 多 件 


何 f 区 网 绍 调 二 


热 接 商 品 : 欣 件 


新 品 上 市 热 销 商品 。 。 推荐 商品 人 气 商 品 


产品 的 效果 。 
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我 的 购物 车 
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| 清空 购物 车 





Cs) 
图 23.6 Go 购 商 城 购物 车 页 面 效 果 图 
当 用 户 到 收银 台 付款 后 ， 系 统 将 自动 生成 订单 ， 会 员 可 通过 单 








[Cae 


f 左 侧 导航 栏 中 的 “我 的 订单 ” 超 


439 


MySQL 从 入 门 到 精通 


链接 查看 自己 的 订单 信息 ， 如 图 23.7 所 示 。 
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图 23.7 Go 购 商 城 会 员 订 单 信息 效果 图 
23.4 数据 库 设 计 


整个 应 用 系统 的 运行 离 不 开 数据 库 的 支持 ， 数 据 库 可 以 说 是 应 用 系统 的 灵魂， 没有 了 数据 库 的 支 
撑 ， 系 统 只 能 说 是 一 个 空 架子 ， 它 将 很 难 完成 与 用 户 之 间 的 交互 。 由 此 可 见 ， 数 据 库 在 系统 中 占有 十 
分 重要 的 地 位 。 本 系统 采用 的 是 MySQL 数据 库 ， 通 过 Hibemate 实现 系统 的 持久 化 操作 。 

本 节 将 根据 Go 购 商 城 网 站 的 核心 实体 类 ， 分 别 设计 对 应 的 E-R 图 和 数据 表 。 





23.4.1 ”数据 库 概念 设计 


所 谓 的 数据 库 概念 化 设计 ， 就 是 将 现实 世界 中 的 对 象 以 E-R 图 的 形式 展现 出 来 ， 本 节 将 对 程序 所 


应 用 到 的 核心 实体 对 象 设计 对 应 的 E-R 图 。 
(1) 会 员 信息 表 tb_customer 的 E-R 图 ， 如 图 23.8 所 示 。 





( 用 户 名 ) 











(编号 





(登录 密码 ) 


地 址 ) 


( 真实 姓名 ) ( 





图 23.8 会 员 信 息 表 也 _customer 的 E-R 图 


(2) 订单 信息 表 tb_order 的 E-R 图 ， 如 图 23.9 所 示 。 


第 23 章 ”Struts 2+Spring+HibematetMySQL 实现 网 络 商城 





订单 名 称 编号 所 属 会 员 iD 
送 货 地 址 订单 订单 状态 
联系 电话 本 2 Te 付款 方式 
商品 价格 创建 时 间 


图 23.9 订单 信息 表 tb_order 的 E-R 图 
(3) 订单 条 目 信息 表 | 的 E-R 图 ， 如 图 23.10 所 示 。 




















(编号 订单 条 目 | ( iiip ) 
( 商品 名 称 ) / \ 、 订 单 iD “iD ) 
( 商品 价格 ) 数量 


23.10 ”订单 条 目 信 息 表 tb_orderitem 的 E-R 图 
(4) 商品 信息 表 tb_productinfo 的 E-R 图 ， 如 图 23.11 所 示 。 


商品 名 称 编号 上 传 图 片 
人 


(ern ) 一 一 人 ea ) 


采购 价格 
市 场 价格 人 
( mimtem ) 

















(ssek ) 


图 23.11 商品 信息 表 tb_productinfo 的 E-R 
(5) 商品 类 别 信息 表 tb_productcategory 的 E-R 图 ， 如 图 23.12 所 示 。 
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i | 上 ND 
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图 23.12 商品 类 别 信息 表 tb_productcategory 的 E-R 
23.4.2 ”创建 数据 库 及 数据 表 


本 系统 采用 MySQL 数据 库 ， 创 建 的 数据 库 名 称 为 db_database24， 数 据 库 db_database24 中 包含 7 
张 数 据 表 。 所 有 数据 表 的 定义 如 下 。 
1. tb_customer (会 员 信息 表 ) 
用 于 存储 会 员 的 注册 信息 ， 该 表 的 结构 如 表 23.1 所 示 。 
表 23.1 tb_customer 信息 表 的 表 结构 


字 段 名 数据 类 型 是 否 为 皇 是 否 主 键 说 明 
id i i 系统 自动 编号 
usemame | 会 员 名 


password VARCHARG50) | 否 | | 。 NULL | 登录 密码 
realname 真实 姓名 


adiress | VARCHARG00) | 是 ”| 盏 | NurL | 地 上 
email | VARCHAR(50) | 是 | 否 | Nurr | 电子 邮件 
mobile | VARCHAR(1) | 是 | 否 | Nurr | 电话 9 码 


2. tb_order (订单 信息 表 ) 
用 于 存储 会 员 的 订单 信息 ， 该 表 的 结构 如 表 23.2 所 示 。 
表 23.2 tb_order 信息 表 的 表 结构 



































字 段 名 数据 类 型 说 明 
id INT(0) 否 是 系统 自动 编号 
name VARCHAR(50) 否 否 订单 名 称 
address VARCHAR(200) 下 否 送 货 地 址 
mobile VARCHAR(11) 理 否 联系 电话 
totalPrice FLOAT 是 否 商品 价格 
createTime DATETIME 是 否 创建 时 间 
paymentWay VARCHAR(15) 是 否 付款 方式 
orderState VARCHAR(10) 是 否 订单 状态 
customerId INT(1) 是 否 会 员 iD 
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3. tb_orderitem 〈 订 单条 目 信息 表 ) 
于 存储 会 员 订 单 的 条 目 信息 ， 该 表 的 结构 如 表 23.3 所 示 。 


表 23.3 tb_orderitem 信息 表 的 表 结构 






































字 段 名 数据 类 型 是 否 为 空 是 否 主键 默 认 值 说 ”有明 
这 | mrao NULL 系统 自动 编号 
productld | mray NULL 商品 训 
productName “| VARCHAR(200) NULL 商品 名 称 












FLOAT 
INT(11) 
VARCHAR(30) 


商品 价格 
商品 数量 


NULL 
NULL 
NULL 


productPrice 
amount 
orderId 













向 | 疝 | 开 | 项 | 玉 | 天 
项 | 天台 | 项 | 节 | 并 





4. tb_productinfo〔 商 品 信息 表 ) 
用 于 存储 商品 信息 ， 该 表 的 结构 如 表 23.4 所 示 。 
表 23.4 tb_productinfo 信息 表 的 表 结 构 














字段 名 数据 类 型 说 了 明 
id INTQO) | 否 | 是 | NUIL | 系统 自动 编 
name VARCHAR(100) NULL 商品 名 称 
description TEXT | 是 | 否 | NULL 商品 描述 
createTime DATETIME | 是 | 否 | NULL | 创建 时 间 
baseprice FLOAT | 是 | 青 | ur | 采购 价格 
marketprice | FLOAT | 是 | 否 | NULL | 市 场 价格 
sellprice FLOAT | 是 | 否 | NULL | 销售 价格 
sexrequest VARCHARG) | 是 | 否 | Nu | 所 属性 别 
commend BIT(1) 是 否 NULL 是 否 推 荐 
clickcount INT(1) 是 否 NULL 浏览 量 
sellCount INTUD) 是 否 NULL 销售 量 
UploadFile INT(1) 是 否 NULL 上 传 图 片 D 








5.tb_productcategory 〈 商 品类 别 信息 表 ) 
用 于 存储 商品 的 类 别 信息 ， 该 表 的 结构 如 表 23.5 所 示 。 


表 23.5 tb_productcategory 信息 表 的 表 结 构 











说 明 
系统 自动 编号 
类 别名 称 
类 别 级 别 
父 节 点 类 别 访 








VARCHAR(50) 
| mray 
INTQD) 
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6. tb_user (管理 
用 于 存储 网 站 后 台 管 理 员 信息 ， 该 表 的 结构 如 表 23.6 所 示 。 
表 23.6 tb_user 信息 表 的 表 结构 


于 


7.tb_uploadfile (上 传 文件 信息 表 ) 
用 于 存储 上 传 文件 的 路 径 信息 ， 该 表 的 结构 如 表 23.7 所 示 。 
表 23.7 tb_uploadfile 信息 表 的 表 结 构 


段 名 | 数据 类 型 说 明 


Pn 
一 
员 
还 
















[mrao 
usemame | VARCHAR(S0) 
VARCHAR(S0) 








password 





| Nu 


[varcHaRGsD | 再 | 是 ”| Nurr | 文人 信和 





23.5 公共 模块 的 设计 


在 项 目 中 经 常会 有 一 些 公共 类 ， 例 如 Hibernate 的 初始 化 类 ， 一 些 自 定义 的 字符 串 处 理 方法 ， 抽 取 
系统 中 公共 模块 更 加 有 利于 代码 重用 ， 同 时 也 能 提高 程序 的 开发 效率 。 在 进行 正式 开发 时 首先 要 进行 
公共 模块 的 编写 。 下 面 介绍 Go 购 商城 的 公共 类 。 


23.5.1 泛 型 工具 类 


Hibemate 提供 了 高 效 的 对 象 到 关系 型 数据 库 的 持久 化 服务 ， 通 过 面向 对 象 的 思想 进行 数据 持久 化 
的 操作 ，Hibernate 的 操作 对 象 就 是 数据 表 所 对 应 的 实体 对 象 , 为 了 将 一 些 公用 的 持久 化 方法 提取 出 来 ， 
首先 需要 实现 获取 实体 对 象 的 类 型 方法 ， 在 本 应 用 中 通过 自 定义 创建 一 个 泛 型 工具 类 GenericsUtils 来 
达到 此 目的 ， 其 代码 如 下 。 


代码 位 置 ， 光盘 \TM\sl23\Shop\src\com\lyq\util\HibemateUtils 
public class GenericsUtils { 
@SuppressWarnings("unchecked") 
public static Class getGenericType(Class clazzX{ 
Type genType = clazz.getGenericSuperclass(); /得 到 泛 型 父 类 
Typel types = ((ParameterizedType) genType).getActualTypeArguments(); 
if (ltypes[0] instanceof Class)) { 
return Object.class; 





} 
return (Class) types[0]; 
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} 

// 获 取 对 象 的 类 名 称 
@SuppressWarnings("unchecked") 

public static String getGenericName(Class clazz}{ 
return clazz.getSimpleName(); 


} 


23.5.2 ”数据 持久 化 类 





在 本 应 用 中 利用 DAO 模式 实现 数据 库 基 本 操作 方法 的 封装 , 数据 库 中 最 为 基本 的 操作 就 是 增 、 删 、 
改 、 查 ， 据 此 自 定 义 数据 库 操作 的 公共 方法 。 由 控制 器 负责 获取 请 求 参数 并 控制 转发 ， 由 DAOSupport 
类 组 织 SQL 语句 。 

根据 自 定义 的 数据 库 操作 的 公共 方法 ， 创 建 接口 BaseDao<T>， 关 键 代码 如 下 。 

代码 位 置 : 光盘 \TM\sl23\Shop\src\com\lyq\dao\BaseDao 

public interface BaseDao<T> { 





// 基 本 数据 库 操作 方法 

public void save(Object obj); /保存 数据 

public void saveOrUpdate(Object obj); /保存 或 修改 数据 
public void update(Object obj); /修改 数据 

public void delete(Serializable ... ids); /删除 数据 

public T get(Serializable entityld); 1/ 加 载 实体 对 象 
public T load(Serializable entityld); 1/ 加载 实体 对 象 
public Object uniqueResult(String hql, Object0 queryParams); /使 用 hql 语句 操作 


创建 类 DaoSupport， 该 类 继承 BaseDao<T> 接 口 ， 在 类 中 实现 接口 中 自 定义 的 方法 ， 其 关键 代码 


如 下 。 


代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\dao\DaoSupport 
public abstract class DaoSupport<T> implements BaseDao<T>{ 


protected Class<T> entityClass = GenericsUtils.getGenericType(this.getClass()); ” // 泛 型 的 类 型 
@Autowired 

protected HibernateTemplate template; /Hibernate 模板 
public HibernateTemplate getTemplate() { 

return template; 


} 

// 删 除 指定 的 对 象 信息 

public void delete(Serializable ... ids) { 

for (Serializable id : ids) { /遍历 标识 参数 
Tt= (T) getTemplate().load(this.entityClass, id); // 加 载 指定 对 象 
getTemplate().delete(t); /执行 删除 操作 
} 


} 
/利用 get() 方 法 加 载 对 象 ， 获 取 对 象 的 详细 信息 
@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true) 
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public T get(Serializable entityld) { 
return (T) getTemplate().get(this.entityClass, entityld); 1/ 加载 指定 对 象 


} 

/利用 load() 方 法 加 载 对 象 ， 获 取 对 象 的 详细 信息 
@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true) 
publicT load(Serializable entityld) { 

return (T) getTemplate().load(this.entityClass, entityld); /加 载 指定 对 象 


} 

/利用 save() 方 法 保存 对 象 的 详细 信息 
public void save(Object obj) { 
getTemplate().save(obj); // 


} 

// 保 存 或 修改 信息 

public void saveOrUpdate(Object obj) { 
getTemplate().saveOrUpdate(ob)); 


} 

/利用 update() 方 法 修改 对 象 的 详细 信息 
public void update(Object obj) { 
getTemplate().update(obj); 

} 


23.5.3 ”分 页 操作 


分 页 查询 是 Java Web 开发 中 十 分 常用 的 技术 。 在 数据 库 量 非常 大 的 情况 下 ， 不 适合 将 所 有 数据 显 


示 到 一 个 页 面 之 中 ， 这 样 既 给 查看 带 来 不 便 ， 又 占用 程序 及 数据 库 的 资 


需要 对 数据 进行 分 





， 此 时 





页 查询 .本 系统 应 用 Hibernate 的 find 方 法 实现 数据 分 页 的 操作 ,将 分 页 的 方法 封装 在 创建 类 DaoSupport 


中 ， 


下 面 将 介绍 Hibemate 分 页 实现 的 方法 。 


1. 分 页 实体 对 象 
首先 定义 分 页 的 实体 对 象 ， 封 装 分 页 基本 属性 信息 和 在 分 页 过 程 中 使 用 的 获取 页 码 的 方法 。 


代码 位 置 ， 光盘 \TM\sl23\Shop\src\com\lyq\model\PageModel<T> 
public class PageModel<T> { 


private int totalRecords; /总 记录 数 
private List<T> list; /| 结果 集 

private int pageNo; /| 当前 页 

private int pageSize; /每 页 显示 多 少 条 
/取得 第 一 页 

public int getTopPageNo() { 

return 1; 

1 

/取得 上 一 页 


public int getPreviousPageNo(){ 
if (pageNo <= 1){ 
return 1; 
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} 

return pageNo -1; 

, 

/取得 下 一 页 

public int getNextPageNo(){ 

if (pageNo >= getTotalPages()){ /如 果 当 前 页 大 于 页 码 
return getTotalPages() == 0 ? 1 : getTotalPages(); // 返 回 最 后 一 页 

} 

return pageNo + 1; 


} 
// 取 得 最 后 一 页 
public int getBottomPageNo() { 


return getTotalPages() == 0 ? 1 : getTotalPages(); // 如 果 总 页 数 为 0 返回 1， 反 之 返回 总 页 数 


} 

// 取 得 总 页 数 

public int getTotalPages() { 

return (totalRecords + pageSize - 1) / pageSize; 
} 

ee // 省 略 的 Setter 和 Getter 方法 


在 页 面 的 实体 对 象 中 ， 封 装 了 几 个 重要 的 页 码 获 取 方 法 ， 即 获取 第 一 页 、 上 一 页 、 
-页 以 及 总 页 数 的 方法 。 
在 取得 上 一 页 页 码 的 方法 getPreviousPagesNo0 中 ， 如 果 当 前 页 的 页 码 数 为 首页 ， 那 
的 页 码 数 为 1。 


在 获取 最 后 一 页 的 方法 getBottomPageNo0 中 ， 通 过 三 目 运算 符 进 行 选择 判断 返回 的 页 码 ， 如 果 总 


页 数 为 0 则 返回 1， 反 之 返回 总 页 面 数 。 当 数据 库 中 没有 任何 信息 的 时 候 ， 总 页 数 为 0。 
2. 实现 自 定义 分 页 方法 


在 公共 接口 中 定义 几 种 不 同 的 分 页 方法 ， 这 些 方法 定义 使 用 了 相同 的 分 页 方法 ， 不 同 的 参数 ， 自 


定义 分 页 方法 关键 代码 如 下 。 


代码 位 置 : 光盘 \TMN\sl\23\Shop\src\com\lyq\dao\BaseDao 
public interface BaseDao<T> { 


/| 基本 数据 库 操作 方法 
// 分 页 操作 

public long getCount(); 1/ 获取 总 信息 数 

public PageModel<T> find(int pageNo, int maxResult); /普通 分 页 操作 
/搜索 信息 分 页 方法 

public PageModel<T> find(int pageNo, int maxResult,String where, Object[ queryParams); 
// 按 指定 条 件 排序 分 页 方法 

public PageModel<T> find(int pageNo, int maxResult,Map<String, String> orderby); 

/ 按 指定 条 件 分 页 和 排序 的 分 页 方法 


public PageModel<T> find(String where, Objectl queryParams, 
Map<String, String> orderby, int pageNo, int maxResult); 
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23.5.4 ”字符 串 工 具 类 


StringUitl 类 中 主要 实现 了 字符 串 与 其 他 数据 类 型 的 转换 ， 例 如 ， 将 日 期 时 间 型 数据 转换 为 指定 格 
式 的 字符 串 ， 处 理 订单 号 码 的 生成 以 及 验证 字符 串 和 浮 点 数 的 有 效 性 。 该 类 中 声明 的 所 有 方法 都 是 静 
态 方法 ， 以 便 在 其 他 类 中 可 以 通过 StringUitl 类 名 直接 调用 。 

在 方法 中 通过 new Date0 方 法 获取 当前 的 系统 时 间 ， 通 过 SimpleDateFormat 的 format0 方 法 将 日 期 
格式 转换 为 指定 的 日 期 格式 。 该 方法 主要 是 在 操作 数据 库 的 时 候 作 为 一 个 有 效 字段 使 用 ， 例 如 订单 的 
创建 日 期 等 ， 其 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\util\StringUitl 

public static String getStringTime(){ 





Date date = new Date(); // 获 取 当 前 系统 时 间 
SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddHHmmssSSSS"); /设置 格式 化 格式 
return sdf format(date): // 返 回 格式 化 后 的 时 间 


} 
23.5.5 “实体 映射 


由 于 本 程序 中 使 用 了 Hibernate 框架 ， 所 以 需要 创建 实体 对 象 并 通过 Hibernate 的 喘 射 文件 将 实体 
对 象 与 数据 库 中 相应 的 数据 表 进 行 关联 。 在 Go 购 商 城中 有 5 个 主要 的 实体 对 象 , 分 别 是 会 员 实体 对 象 、 
订单 实体 对 象 、 订 单条 目 实体 对 象 、 商 品 实体 对 象 以 及 商品 类 别 实体 对 象 。 

1. 实体 对 象 总 体 设 计 


实体 对 象 是 Hibernate 中 非常 重要 的 一 个 环节 ， 因 为 Hibemate 只 有 通过 映射 文件 建立 实体 对 象 与 
数据 库 数据 表 之 间 的 关系 , 才能 进行 系统 的 持久 化 操作 。 在 Go 购 商 城 网 站 中 主要 实体 对 象 及 其 关系 如 


图 23.13 所 示 。 
二 | | 
Orderltem hbm xml 


多 一 对 多 


(| 
a 
[会 员 : customd 。 多 对 一 | 订单 : 一 对 针 商品 - 多 对 一 | 商品 类 别 : 
映射 文件 映射 文件 映射 文件 映射 文件 


Customer hbm xml Order hbm xml Productinfo hbm xml ProductCategory hbm xml 


图 23.13 ”Go 购 商 城主 要 实体 对 象 及 其 关系 


从 图 23.13 中 可 以 看 到 ， 该 项 目 主要 有 5 个 实体 对 象 ， 分 别 是 会 员 实体 对 象 Customer 类 、 订 单 实 
体 对 象 Order 类 、 订 单条 目 实体 对 象 OrderItem 类 、 商 品 实体 对 象 ProductInfo 类 和 商品 类 别 实体 对 象 











| 
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ProductCategory 类 。 

从 中 可 以 看 到 会 员 与 订单 是 一 对 多 的 关系 ， 一 个 会 员 可 以 对 应 多 张 订单 ， 但 是 每 张 订单 只 能 对 应 
一 个 会 员 ， 订单 条目 与 订单 为 多 对 一 的 关系 ， 一 张 订 单 中 可 以 包含 多 个 订单 条 目 ， 但 是 每 个 订单 条 目 
只 能 对 应 一 张 订单 ;订单 与 产品 是 一 对 多 关系 ， 一 张 订单 可 以 对 应 多 个 商品 ; 商品 与 商品 类 别 是 多 对 
一 关系 ， 多 件 商品 可 以 对 应 一 个 商品 类 别 。 

其 中 的 “*.hbm.xml” 文 件 为 实体 对 象 的 Hibernate 映射 文件 。 


2. 会 员 信 息 
Customer 类 为 会 员 信 息 实体 类 ， 用 于 封装 会 员 的 注册 信息 ， 其 关键 代码 如 下 。 


代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\usen\Customer 
public class Customer implements Serializable{ 





private Integer id; /用 户 编号 

private String username; // 用 户 名 

private String password; /密码 

private String realname; // 真 实 姓名 

private String email; /邮箱 

private String address; /住址 

private String mobile; // 手 机 

这 // 省 略 的 Setter 和 Getter 方法 


创建 会 员 信息 实体 类 的 喘 射 文件 Customer.hbm.xml， 在 映射 文件 中 配置 会 员 实体 类 属性 与 数据 表 
tb_customer 响应 字段 的 关联 ， 并 声明 用 户 编 号 的 主键 生成 策略 为 自动 增长 ， 配 置 文 件 中 的 关键 代码 
如 下 。 


代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\user\Customer.hbm.xml 
<?xml version="1.0" encoding="UTF-8"?> 
<IDOCTYPE hibernate-mapping PUBLIC 
"-//Hibernate/Hibernate Mapping DTD 3.0//EN" 
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd" > 
<hibernate-mapping package="com.lyq.model.user"> 
<class name="Customer" table="tb_customer"> 
<id name="id"> 
<generator class="native"/> 
</id> 
<property name="username'" not-null="true" length="50"/> 
<property name="password" not-null="true" length="50"/> 
<property name="realname" length="20"/> 
<property name="address" length="200"/> 
<property name="email" length="50"/> 
<property name="mobile" length="11"/> 
</class> 
</hibernate-mapping> 
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3. 订单 信息 


Order 类 为 订单 信息 实体 类 ,用 户 封装 订单 的 基本 信息 ， 但 是 不 包括 详细 的 订购 信息 ， 其 关键 代码 
如 下 。 
代码 位 置 : 光盘 \TMN\sl23\Shop\src\com\lyqumodel\order\Order 


public class Order implements Serializable { 


private String orderld; /订单 编号 (手动 分 配 ) 
private Customer customer; /所 属 用 户 
private String name; /路 货 人 姓名 
private String address; /路 货 人 住址 
private String mobile; /路 货 人 手机 
private Set<Orderltem> orderltems; /所 买 商品 
private Float totalPrice; /| 总额 
private PaymentWay paymentWay: /支付 方式 
private OrderState orderState; // 订 单 状 态 
private Date createTime = new Date(); /创建 时 间 
// 省 略 的 Setter 和 Getter 方法 


于 
创建 订单 信息 实体 类 的 映射 文件 Orderhbm.xml， 在 映射 文件 中 配置 订单 实体 类 属性 与 数据 表 


tb_order 字段 的 关联 ， 声 明 主键 orderId 的 主键 生成 策略 为 手动 分 配 ， 并 配置 订单 与 会 员 的 多 对 一 关系 ， 
订单 与 订单 项 的 一 对 多 关系 ， 其 关键 代码 如 下 。 
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代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\model\order\Order 
<?xml version="1.0" encoding="UTF-8"?> 
<IDOCTYPE hibernate-mapping PUBLIC 
"-//Hibernate/Hibernate Mapping DTD 3.0//EN" 
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd" > 
<hibernate-mapping package="com.lyq.model.order"> 
<class name="Order" table="tb_order"> 
<id name="orderld" type="string" length="30"> 
<generator class="assigned"/> 
</id> 
<property name="name" not-null="true" length="50"/> 
<property name="address" not-null="true" length="200"/> 
<property name="mobile" not-null="true" length="11"/> 
<property name="totalPrice"/> 
<property name="createTime" /> 
<property name="paymentWay" type="com.lyq.util.hibernate.PaymentWayType" length="15"/> 
<property name="orderState" type="com.lyq.util.hibernate.OrderStateType" length="10"/> 
< 上 - 多 对 一 映射 用 户 --> 
<many-to-one name="customer" column="customerld"/> 
<!-- 映射 订单 项 --> 
<set name="orderltems" inverse="true" lazy="extra" cascade="all"> 
<key column="orderld"/> 
<one-to-many class="Orderltem"/> 
</set> 
</class> 
</hibernate-mapping> 
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4. 订单 条 目 信息 
OrderItem 类 为 订单 条 目的 实体 对 象 ， 用 于 封装 一 个 订单 中 的 一 条 详细 商品 采购 信息 ， 其 关键 代码 
如 下 。 


代码 位 置 : 光盘 \TM\sl\23\Shop\srcvcomlyqmodelvorderOrderItem 
public class Orderltem implements Serializable{ 





private Integer id; /商品 条 目 编 号 

private Integer productld; /商品 id 

private String productName; // 商 品名 称 

private Float productMarketprice; /市 场 价格 

private Float productPrice; /商品 销售 价格 

private Integer amount=1; // 购 买 数 量 

private Order order; /所 属 订单 iD 

a // 省 略 的 Setter 和 Getter 方法 


} 


创建 订单 条 目 信息 实体 类 的 映射 文件 OrderItem.hbm.xml， 在 映射 文件 中 配置 订单 条 目 实体 类 属性 
与 数据 表 tb_orderitem 字段 的 关联 ， 声 明 主键 id 的 主键 生成 策略 为 自动 增长 ， 并 配置 订单 条 目 与 订单 
的 多 对 一 关系 ， 其 关键 代码 如 下 。 
代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\model\order\OrderItem.hbm.xml 
<?xml version="1.0" encoding="UTF-8"?> 
<IDOCTYPE hibernate-mapping PUBLIC 
"-//Hibernate/Hibernate Mapping DTD 3.0//EN" 


"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd" > 
<hibernate-mapping package="com.lyq.model.order"> 





<generator class="native"/> 
</id> 


<property name="productld" not-null="true"/> 
<property name="productName" not-null="true" length="200"/> 
<property name="productPrice" not-null="true"/> 
<property name="amount"/> 
< 上 - 多 对 一 映射 订单 -> 
<many-to-one name="order" column="orderld"/> 
</class> 
</hibernate-mapping> 


5. 商品 信息 


ProductInfo 类 为 商品 信息 实体 类 ， 主 要 用 于 封装 商品 相关 的 基本 信息 ， 它 是 整个 系统 中 最 为 重要 
的 一 个 实体 对 象 ， 也 是 应 用 最 多 的 一 个 实体 对 象 ， 整 个 网 站 的 业务 流程 都 以 商品 为 核心 进行 展开 ， 其 
关键 代码 如 下 。 

代码 位 置 : 光盘 \TM\sl\23\Shop\src\com\lyqmodel\product\ProductInfo 


public class Productlnfo implements Serializable { 
private Integer id; // 商 品 编号 




















眶 
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private String name; /商品 名 称 
private String description; /商品 说 明 
private Date createTime = new Date(); // 上 架 时 间 
private Float baseprice; /商品 采 购 价格 
private Float marketprice; /现在 市 场 价格 
private Float sellprice; /商城 销售 价格 
private Sex sexrequest ; /所 属性 别 
private Boolean commend = false; // 是 否 是 推荐 商品 默认 值 为 false) 
private Integer clickcount = 1; // 访 问 量 《统计 受 欢 迎 的 程度 ) 
private Integer sellCount = 0; // 销 售 数 量 (统计 热 销 商品 ) 
private ProductCategory category; // 所 属 类 别 
private UploadFile uploadFile; /上 传 文件 
过 // 省 略 的 Setter 和 Getter 方法 
册 
创建 商品 信息 实体 类 的 映射 文件 ProductInfo.hbm.xml， 在 映射 文件 中 配置 商品 实体 类 属性 与 数据 





表 tb_productinfo 字段 的 关联 , 并 声明 其 主键 id 的 生成 策略 为 自动 增长 , 并 配置 商品 与 商品 类 别 多 对 一 
关联 关系 、 商 品 与 商品 上 传 文件 的 多 对 一 关联 关系 ， 其 关键 代码 如 下 。 
代码 位 置 ， 光 盘 \TM\sl\23\Shop\src\com\lyq\model\product\ProductInfo.hbm.xml 
<?xml version="1.0" encoding="UTF-8"?> 
<IDOCTYPE hibernate-mapping PUBLIC 
"-//Hibernate/Hibernate Mapping DTD 3.0//EN" 
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd" > 
<hibernate-mapping package="com.lyq.model.product"> 
<class name="Productlnfo" table="tb_productlnfo"> 
<id name="id"> 
<generator class="native"/> 
</id> 
<property name="name" not-null="true" length="100"/> 
<property name="description" type="text"/> 
<property name="createTime"/> 
<property name="baseprice"/> 
<property name="marketprice"/> 
<property name="sellprice"/> 
<property name="sexrequest" type="com.lyq.util.hibernate.SexType" length="5"/> 
<property name="commend"/> 
<property name="clickcount"/> 
<property name="sellCount"/> 
<!-- 多 对 一 映射 类 别 --> 
<many-to-one name="category" column="categoryld"/> 
<I-- 多 对 一 映射 上 传 文件 一 > 
<many-to-one name="uploadFile" unique="true" cascade="all" lazy="false"/> 
</class> 
</hibernate-mapping> 


6. 商品 类 别 信息 
ProductCategory 类 为 商品 类 别 的 实体 对 象 ， 主要 用 于 封装 商品 类 别 的 基本 信息 ， 其 关键 代码 如 下 。 
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代码 位 置 : 光盘 \TMVsl\23\Shop\srcvcomlyqmodel\productProductCategory 
public class ProductCategory implements Serializable { 


private Integer id; /| 类 别 编号 
private String name; /| 类 别名 称 
private int level = 1; /层次 
private Set<ProductCategory> children; // 子 产品 类 别 
private ProductCategory parent; 1/ 父 类 别 
private Set<ProductInfo> products = new TreeSet<Productinfo>();”// 包 含 商品 
// 省 略 的 Setter 和 Getter 方法 


} 


创建 商品 类 别 信息 实体 类 的 映射 文件 ProductCategory.hbm.xml， 在 映射 文件 中 配置 商品 类 别 实体 
类 属性 与 数据 表 tb_productcategory 字段 的 关联 ， 并 配置 商品 类 别 与 商品 的 一 对 多 的 关联 关系 ， 商 品 
别 与 其 父 节 点 多 对 一 的 关联 关系 ， 商 品类 别 与 其 子 节点 一 对 多 的 关联 关系 ， 其 关键 代码 如 下 。 
代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\model\product\ProductCategory.hbm.xml 
<?xml version="1.0" encoding="UTF-8"?> 
<IDOCTYPE hibernate-mapping PUBLIC 
"-//Hibernate/Hibernate Mapping DTD 3.0//EN" 
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd" > 
<hibernate-mapping package="com.lyq.model.product"> 
<class name="ProductCategory" table="tb_productCategory"> 
<id name="id"> 
<generator class="native"/> 
</id> 
<property name="name" not-null="true" length="50"/> 
<property name="level"/> 
<!-- 映射 包含 的 商品 集合 一 > 
<set name="products" inverse="true" lazy="extra"> 
<key column="categoryld"/> 
<one-to-many class="Productlnfo" /> 
</set> 
<!- 映射 父 节点 --> 
<many-to-one name="parent" column="pid"/> 
<!- 映射 子 节点 --> 
<set name="children" inverse="true" lazy="extra" cascade="all" order-by="id"> 
<key column="pid"/> 
<one-to-many class="ProductCategory"/> 
</set> 
</class> 
</hibernate-mapping> 


23.6 项 目 环境 搭建 
项 目 正 式 开 发 的 第 一 步 就 是 搭建 项 目的 环境 以 及 项 目 集成 的 框架 等 ， 都 说 万 丈 高 楼 平地 起 ， 从 此 
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开始 将 踏 上 万 里 征程 的 第 一 步 ， 在 此 之 前 需要 将 Spring、Stmuts 2、Hibemate 以 及 系统 应 用 的 其 他 jar 
包 导 入 项 目的 lib 文件 下 。 


23.6.1 配置 Struts 2 


struts.xml 文件 是 Struts 2 重要 的 配置 文件 ， 通 过 对 该 文件 的 配置 实现 程序 的 Action 与 用 户 请 求 之 
间 的 映射 、 视 图 映射 等 重要 的 配置 信息 。 在 项 目的 ClassPath 下 创建 struts.xml 文件 ， 其 配置 代码 如 下 。 


代码 位 置 ， 光盘 \TM\sl\23\Shop\src\struts.xml 
<?xml version="1.0" encoding="UTF-8"?> 
<!IDOCTYPE struts PUBLIC 
"-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" 
"http://struts.apache.org/dtds/struts-2.1.dtd" > 
<struts> 
<!- 前 后 台 公共 视图 的 映射 一 > 
<include file="com/lyq/action/struts-default.xm!l" /> 
<!- 后 台 管 理 的 Struts 2 配置 文件 --> 
<include file="com/lyq/action/struts-admin.xml" /> 
<!-- 前 台 台 管理 的 Struts 2 配置 文件 --> 
<include file="com/lyq/action/struts-front.xml" /> 
</struts> 


为 了 便于 程序 的 维护 和 管理 ， 将 前 后 台 的 Struts 2 配置 文件 进行 分 开 处 理 ， 然 后 通过 include 标签 
加 载 在 系统 默认 加 载 的 Struts 2 配置 文件 中 。 在 此 将 Struts 2 配置 文件 分 为 3 个 部 分 ，struts-default.xml 
文件 为 前 后 台 公共 的 视图 映射 配置 文件 ， 其 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\struts-default.xml 

<?xml version="1.0" encoding="UTF-8" ?> 

<IDOCTYPE struts PUBLIC 

"-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" 
"http://struts.apache.org/dtds/struts-2.1.dtd"> 
<struts> 
<!- OGNL 可 以 使 用 静态 方法 -> 
<constant name="struts.ognl.allowStaticMethodAccess" value="true"/> 
<package name="shop-default" abstract="true" extends="struts-default"> 
<global-results> 
<!-- 省 略 的 配置 信息 --> 
</global-results> 
<global-exception-mappings> 
<exception-mapping result="error" 
exception="com.lyq.util. AppException"></exception-mapping> 
</global-exception-mappings> 
</package> 
</struts> 


后 台 管 理 的 Struts 2 配置 文件 struts-admin.xml 主要 负责 后 台 用 户 请 求 的 Action 和 视图 映射 ， 其 代 
码 如 下 。 
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代码 位 置 : 光盘 \TMN\sl23\Shop\src\com\lyq\action\struts-admin.xml 


<?xml version="1.0" encoding="UTF-8"?> 
<IDOCTYPE struts PUBLIC 
"-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" 
"http://struts.apache.org/dtds/struts-2.1.dtd" > 
<struts> 
<!- 后 台 管理 -> 
<package name="shop.admin" namespace="/admin" extends="shop-default"> 
<!-- 配置 拦截 器 --> 
<interceptors> 
<!-- 验证 用 户 登录 的 拦截 器 一 > 
<interceptor name="loginInterceptor" 
class="com.lyq.action.interceptor.UserLogininterceptor"/> 
<interceptor-stack name="adminDefaultStack"> 
<interceptor-ref name="loginInterceptor"/> 
<interceptor-ref name="defaultStack"/> 
</interceptor-stack> 
</interceptors> 
<action name="admin_*" class="indexAction" method="{1}"> 
<result name="top">/WEB-INF/pages/admin/top.jsp</result> 
.…<!-- 省 略 的 Action 配置 --> 
<interceptor-ref name="adminDefaultStack"/> 
</action> 


</package> 
<package name="shop.admin.user namespace="/admin/user" extends="shop-default"> 
<action name="user_” method="{1}" class="userAction"></action> 
</package> 
<!- 栏目 管理 --> 
<package name="shop.admin.category" namespace="/admin/product" extends="shop.admin"> 
<action name="category_*" method="{1}" class="productCategoryAction"> 
.…<!- 省 略 的 Action 配置 --> 
<interceptor-ref name="adminDefaultStack"/> 





</action> 
</package> 
<!-- 商品 管理 -> 


<package name="shop.admin.product" namespace="/admin/product" extends="shop.admin"> 
<action name="product_*" method="{1}" class="productAction"> 
.< 上 -省 略 的 Action 配置 -> 
<interceptor-ref name="adminDefaultStack"/> 


</action> 
</package> 
< 上 - 订单 管理 一 > 


<package name="shop.admin.order" namespace="/admin/product" extends="shop.admin"> 
<action name="order_*" method="{1}" class="orderAction"> 
-…<!-- 省 略 的 Action 配置 --> 
<interceptor-ref name="adminDefaultStack"/> 
</action> 
</package> 
</struts> 
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前 台 管理 的 Struts 2 配置 文件 stmts-ffontxml 主要 负责 后 台 用 户 请 求 的 Action 和 视图 映射 , 其 代码 





如 下 。 


代码 位 置 : 光盘 \TMVsl\23\Shop\srcvcomlyqvaction\struts-ffont xml 
<?xml version="1.0" encoding="UTF-8"?> 
<IDOCTYPE struts PUBLIC 
"-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" 
"http://struts.apache.org/dtds/struts-2.1.dtd" > 
<struts> 
<!-- 程序 前 台 --> 
<package name="shop.front" extends="shop-default"> 
<!-- 配置 拦截 器 --> 
<interceptors> 
<!-- 验证 用 户 登录 的 拦截 器 -> 
<interceptor name="|loginInterceptor" 
class="com.lyq.action.interceptor.CustomerLogininteceptor"/> 
<interceptor-stack name="customerDefaultStack"> 
<interceptor-ref name="loginInterceptor"/> 
<interceptor-ref name="defaultStack"/> 
</interceptor-stack> 
</interceptors> 
<action name="index" class="indexAction"> 
<result>/WEB-INF/pages/index.jsp</result> 
</action> 
</package> 
<!-- 消费 者 Action --> 
<package name="shop.customer" extends="shop-default" namespace="/customer"> 
<action name="customer_*" method="{1}" class="customerAction"></action> 
</package> 
<!-- 商品 Action --> 
<package name="shop.product" extends="shop-default" namespace="/product"> 
<action name="product_*" class="productAction" method="{1}"> 
.…<!- 省 略 的 Action 配置 --> 
</action> 
</package> 
<!-- 购物 车 Action --> 
<package name="shop.cart" extends="shop.front" namespace="/product"> 
<action name="cart_*" class="cartAction" method="{1}"> 
.…<!-- 省 略 的 Action 配置 --> 
<interceptor-ref name="customerDefaultStack"/> 
</action> 
</package> 
<!-- 订单 Action --> 
<package name="shop.order" extends="shop.front" namespace="/product"> 
<action name="order_*" class="orderAction" method="{1}"> 
.<|-- 省 略 的 Action 配置 --> 
<interceptor-ref name="customerDefaultStack"/> 
</action> 
</package> 
</struts> 
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Struts 2 与 Struts 1 是 两 个 完全 不 同 的 开发 框架 ， 除 其 核心 控制 器 不 同 以 外 ， 还 有 以 下 3 个 不 同 点 。 
1. 命名 空间 的 不 同 


在 Struts 2 中 的 Filter 默认 的 扩展 名 为 .action, 这 是 在 defaultproperties 文件 的 struts.action.extension 
属性 中 定义 的 ;而 Struts 1 则 通过 <initpattermn> 属 性 来 配置 。Struts 1 与 Struts 2 两 个 框架 的 命名 空间 不 
一 样 ， 因 此 Stmts 1 和 Struts 2 可 以 在 同一 个 Web 应 用 系统 中 无 障碍 地 共存 。 


2. 设置 系统 属性 的 不 同 

Stmts 2 的 配置 信息 中 不 需要 通过 <init-param> 来 设置 系统 的 属性 , 它 并 不 是 取消 了 这 些 属性 , 而 是 
使 用 default.properties 文件 作为 默认 的 配置 选项 文件 ， 并 可 以 通过 struts.properties 的 文件 设置 不 同 的 属 
性 值 来 覆盖 默认 文件 的 值 实现 自己 的 配置 。 

3. 映射 文件 名 的 配置 参数 的 不 同 

在 Stmts 2 中 没有 提供 映射 文件 名 的 配置 参数 ， 取 而 代 之 的 是 默认 配置 文件 struts.xml。 

















23.6.2 配置 Hibernate 


Hibernate 配置 文件 主要 用 于 配置 数据 库 连接 和 Hibemate 运行 时 所 需 的 各 种 属性 , 这 个 配置 文件 位 
于 应 用 程序 或 Web 程序 的 类 文件 夹 classes 中 。Hibernate 配置 文件 支持 两 种 形式 ， 一 种 是 XML 格式 的 
配置 文件 ， 另 一 种 是 Java 属性 文件 格式 的 配置 文件 ， 采 用 “ 键 = 值 ”的 形式 。 建 议 采 用 XML 格式 的 配 
置 文件 。 

在 Hibernate 的 配置 文件 中 配置 连接 的 数据 库 的 连接 信息 ， 数 据 库 方言 以 及 打印 SQL 语句 等 属性 ， 
其 关键 代码 如 下 。 


代码 位 置 ， 光盘 \TM\sl\23\Shop\src\hibernate.cfe.xml 
<?xml version="1.0" encoding="UTF-8"?> 
<IDOCTYPE hibernate-configuration PUBLIC 
"-//Hibernate/Hibernate Configuration DTD 3.0//EN" 
"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd" > 
<hibernate-configuration> 
<session-factory> 
<!-- 数据 库 方言 --> 
<property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property> 
<!-- 数据 库 驱动 -> 
<property name="hibernate.connection.driver_class">com.mysqljdbc.Driver</property> 
< 上 -- 数据 库 连 接 信息 一 > 
<property name="hibernate.connection.url"> 
jdbc:mysql://localhost:3306/db_database24</property> 
<property name="hibermnate.connection.username">root</property> 
<property name="hibernate.connection.password">111</property> 
<!-- 打印 SQL 语句 一 > 
<property name="hibernate.show_sql">true</property> 
<!-- 不 格式 化 SQL 语句 -> 
<property name="hibermnate .format_sql">false</property> 
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<!- 为 Session 指定 一 个 自 定义 策略 一 > 
<property name="hibernate.current_session_context_class">thread</property> 
<!-- C3P0 JDBC 连接 池 --> 
<property name="hibernate.c3p0.max_size">20</property> 
<property name="hibernate.c3p0.min_size">5</property> 
<property name="hibernate.c3p0.timeout">120</property> 
<property name="hibernate.c3p0.max_statements">100</property> 
<property name="hibernate.c3p0.idle_test_period">120</property> 
<property name="hibernate.c3p0.acquire_increment">2</property> 
<property name="hibernate.c3p0.validate">true</property> 
<!-- 映射 文件 --> 
<mapping resource="com/lyq/model/user/User.hbm.xml"/> 
-…<!- 省 略 的 映射 文件 -> 
</session-factory> 
</hibernate-configuration> 


0 涪 田 

C3P0 是 一 个 随 Hibernate 一 同 分 发 的 开发 的 JDBC 连接 池 , 它 位 于 Hibernate 源 文件 的 lib 目录 
下 。 如 果 在 配置 文件 中 设置 了 hibernate.c3p0.* 的 相关 属性 ，Hibernate 将 会 使 用 C3P0Connection 
Provider 来 缓存 JDBC 连接 。 


23.6.3 配置 Spring 


利用 Spring 加 载 Hibernate 的 配置 文件 以 及 Session 管理 类 ， 所 以 在 配置 Spring 的 时 候 ， 只 需要 
置 Spring 的 核心 配置 文件 applicationContext-common.xml， 其 代码 如 下 。 


代码 位 置 ， 光 盘 \TM\sl\23\Shop\src\applicationContext-common.xml 
<?xml version="1.0" encoding="UTF-8"?> 
<beans xmIns="http://www.springframework.org/schema/beans" 
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
xmlns:context="http://www.springframework.org/schema/context" 
xmlns:aop="http://www.springframework.org/schema/aop” 
xmlns:tx="http://www.springframework.org/schematx" 
xsi:schemaLocation="http://www.springframework.org/schema/beans 
http://www.springframework.org/schema/beans/spring-beans-2.19.xsd 
http://www.springframework.org/schema/context 
http://www.springframework.org/schema/context/spring-context-2.19.xsd 
http://www.springframework.org/schema/aop 
http://www.springframework.org/schema/aop/spring-aop-2.19.xsd 
http://www.springframework.org/schema/tx 
http://www.springframework.org/schema/tx/spring-tx-2.19.xsd"> 
<context:annotation-config/> 
<context:component-scan base-package="com.lyq"/> 
<!-- 配置 sessionFactory --> 
<bean id="sessionFactory" 
class="org.springframework.orm.hibernate3.LocalSessionFactoryBean"> 
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<property name="configLocation"> 
<value>classpath:hibernate.cfg.xml</value> 
</property> 
</bean> 
<!-- 配置 事务 管理 器 --> 
<bean id="transactionManager" 
class="org.springframework.orm.hibernate3.HibernateTransactionManager"> 
<property name="sessionFactory"> 
<ref bean="sessionFactory" /> 
</property> 
</bean> 
<tx:annotation-driven transaction-manager="transactionManager" /> 
<!-- 定义 Hibernate 模板 对 象 -> 
<bean id="hibernateTemplate" class="org.springframework.orm.hibernate3.HibernateTemplate"> 
<property name="sessionFactory" ref="sessionFactory"/> 
</bean> 
</beans> 


23.6.4 配置 web.xml 


任何 MVC 框架 都 需要 与 Servlet 应 用 整合 , 而 Servlet 则 必须 在 web.xml 文件 中 进行 配置 。web.xml 
的 配置 文件 是 项 目的 基本 配置 文件 ,通过 该 文件 设置 实例 化 Spring 容器 、 过 滤器 、 配 置 Struts 2 以 及 设 
置 程序 默认 执行 的 操作 ， 其 关键 代码 如 下 。 


代码 位 置 : 光盘 \TM\sl\23\Shop\WebContent\WEB-INFVweb.xml 

<?xml version="1.0" encoding="UTF-8"?> 

<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
xmlns="http:/Wiava.sun.com/xmlns/javaee" 
xmlns:web="http:Wijava.sun.com/xml/ns/javaee/web-app_2_19.xsd" 
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee 
http://java.sun.com/xml/ns/javaee/web-app_2_19.xsd" 
id="WebApp_ID" version="2.5"> 
<display-name>Shop</display-name> 
<!-- 对 Spring 容器 进行 实例 化 --> 
<listener> 
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> 
</listener> 
<context-param> 
<param-name>contextConfigLocation</param-name> 
<param-value>classpath:applicationContext-*.xml</param-value> 
</context-param> 
<!-- OpenSessionlnViewFilter 过 滤器 --> 
<filter> 
<filter-name>openSessionInViewFilter</filter-name> 
<filter-class>org.springframework.orm.hibernate3.support.OpenSessionInViewFilter</filter-class> 
</filter> 
<filter-mapping> 
<filter-name>openSessionInViewFilter</filter-name> 
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<url-pattern>/*</url-pattern> 
</fiter-mapping> 
<! 一 Struts 2 配置 -> 
<filter> 
<filter-name>struts2</filter-name> 
<filter-class>org.apache.struts2.dispatcher.ng filter. StrutsPrepareAndExecuteFilter</filter-class> 
</filter> 
<filter-mapping> 
<filter-name>struts2</filter-name> 
<url-pattern>/*</url-pattern> 
</filter-mapping> 
<!-- 设置 程序 的 默认 欢迎 页 面 -> 
<welcome-file-list> 
<welcome-file>index.jsp</welcome-file> 
</welcome-file-list> 
</web-app> 


23.7 登录 注册 模块 设计 


如 果 要 提高 网 站 的 安全 性 ， 防 止 非法 用 户 进入 网 站 ， 可 以 让 用 户 进入 网 站 前 先进 行 注册 ， 注 册 成 
功 的 用 户 才 可 以 通过 购物 车 购买 商品 。 用 户 注册 在 大 多 数 网 站 中 都 是 不 可 缺少 的 功能 ， 也 是 用 户 参 与 
网 站 活动 最 为 直接 的 桥梁 。 通 过 用 户 注册 ， 可 以 有 效 地 对 用 户 信息 进行 采集 ， 并 将 合法 的 用 户 信息 保 
存 到 指定 的 数据 表 中 ， 通 常情 况 下 ， 当 用 户 注册 操作 完毕 ， 将 直接 登录 该 网 站 。 


23.7.1 ”模块 概述 
由 于 Go 购 商 城 分 为 前 台 和 后 台 两 个 部 分 , 所 以 登录 也 分 为 前 台 登 录 和 后 台 登 录 两 个 部 分 功能 , 前 


台 的 登录 针对 的 是 在 Go 购 商 城 注册 的 会 员 , 后 台 登 录 主 要 针对 的 是 网 站 管理 员 , 而 注册 模块 主要 针对 
的 就 是 前 台 想 进行 购物 的 游客 ， 如 图 23.14 所 示 。 

















登录 注册 模块 
前 后 前 
台 台 
登 登 注 
录 录 册 
网 站 会 员 模 模 模 网 站 游客 
块 块 块 























网 站 管理 员 


图 23.14 ”登录 注册 模块 框架 图 
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23.7.2 ”注册 模块 的 实现 


在 安全 注册 与 登录 操作 过 程 中 ， 主 要 是 将 表单 内 容 进行 严格 的 校 验 ， 这 样 可 以 提高 网 站 的 安全 性 ， 
防止 非法 用 户 进入 网 站 。 

在 本 模块 中 ， 用 户 注册 页 面 为 customer regjsp， 如 图 
23.15 所 示 ， 用 户 注册 主要 包括 用 户 名 、 密 码 、 确 认 密 码 、 
邮箱 地 址 、 住 址 以 及 手机 号 码 6 个 表单 控件 。 其 中 ， 用 户 
名 要 求 5 一 32 个 字符 , 密码 表单 与 确认 密码 表单 必须 一 致 ; 
邮箱 地 址 表单 必须 是 正确 的 地 址 ， 这 里 通过 Struts 2 的 校 
验 器 进行 校 验 ; 住址 与 手机 号 码 两 个 表单 主要 是 在 用 户 购 
买 商品 生成 订单 时 直接 获取 相关 的 送 货 信息 ， 方 便 用 户 的 
操作 。 

当 用 户 成 功 注册 信息 后 ， 就 可 以 进行 登录 操作 ， 并 在 
Go 购 商 城中 购买 商品 。 














1. 表单 验证 图 23.15 会 员 注册 页 面 
在 Web 开发 中 ， 为 了 确保 数据 的 安全 性 ， 通 常情 况 下 都 会 对 页 面 提交 的 数据 进行 统一 验证 ， 从 而 
保障 数据 的 合法 性 。 


在 Struts 2 中 有 两 种 表单 的 验证 方式 ， 在 本 模块 中 将 使 用 XML 文件 对 表单 中 的 信息 进行 合法 性 验 
证 ， 利 用 requiredsting 校 验 器 对 CustomerAction 类 中 的 字段 进行 非 空 验证 ; 利用 stringlength 校 验 器 对 
CustomerAction 类 中 的 字段 长 度 进行 验证 ， 利 用 email 校 验 器 对 邮箱 地 址 的 格式 进行 验证 ， 在 
CustomerAction 类 的 包 下 新 建 XML 文件 。 

2. 保存 注册 信息 

当 用 户 单 击 会 员 注 册页 面 中 的 “注册 ” 超 链 接 时 ， 系 统 将 会 发 送 一 个 customer reg.html 的 URL 请 
求 ， 该 请 求 将 会 执行 CustomerAction 类 中 的 save0 方 法 ， 在 该 方法 中 首先 判断 用 户 名 是 否 可 用 ， 如 果 
可 用 就 将 注册 信息 保存 在 数据 库 中 ， 否 则 返回 错误 信息 ， 其 关键 代码 如 下 。 

代码 位 置 : 光盘 \TM\sl\23\Shop\src\com\lyq\action\ser\CustomerAction 


public String save() throws Exception{ 
boolean unique = customerDao.isUnique(customer.getUsername()); 1/ 判断 用 户 名 是 否 可 用 








if(uniqueX{ // 如 果 用 户 名 可 用 
customerDao.save(customer); /保存 注册 信息 

return CUSTOMER_LOGIN; /返回 会 员 登 录 页 面 
Jelse{ 

throw new AppException(" 此 用 户 名 不 可 用 "); // 否 则 返回 页 面 错 误 信息 


} 
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23.8 前 台 商 品 信息 查询 模块 设计 


商品 是 Go 购 商 城 的 灵魂 ,只 有 好 的 商品 展示 以 及 丰富 的 商品 信息 才能 吸引 顾客 的 眼球 ,提高 网 站 
的 关注 度 , 这 也 是 为 企业 创造 效益 的 决定 性 因素 , 所 以 Go 购 商城 的 前 台 商 品 展示 在 整个 系统 中 占有 非 
常 重要 的 地 位 。 





23.8.1 模块 概述 


根据 前 台 的 页 面 设计 将 前 台 商 品 信息 查询 模块 划分 为 5 个 模块 ， 主 要 包括 商品 分 类 查询 、 人 气 商 
品 查询 、 热 销 商品 查询 、 推 荐 商品 查询 以 及 商品 的 模糊 查询 ， 如 图 23.16 所 示 。 


前 台 商品 查询 模块 








23.16 前台 商品 信息 查询 模块 框架 图 


23.8.2 ”前 台 商 品 信息 查询 模块 技术 分 析 





在 前 台 的 首页 商品 展示 中 ， 首 先 展现 给 用 户 的 就 是 商品 类 别 的 分 级 显示 ， 方 便 用 户 按 类 别 对 商品 
进行 查询 ， 同 时 也 能 体现 出 Go 购 商 城 产 品种 类 的 丰富 多 样 。 

实现 商品 类 别 的 分 级 查询 ， 首 先 需要 查询 所 有 的 一 级 节点 ， 通 过 公共 模块 持久 化 类 中 封装 的 fmnd0 
方法 实现 该 功能 ， 在 首页 的 Action 请 求 IndexAction 的 execute0 方 法 中 ， 调 用 封装 的 find0 方 法 ， 其 关 
键 代码 如 下 。 

代码 位 置 : 光盘 \TM\sl23\Shop\src\com\lyq\action\IndexAction 


public String execute() throws Exception { 
/查询 所 有 类 别 
String where = "where parent is null"; 
categories = categoryDao .find(-1, -1, where, null).getList(); 
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/省略 的 Setter 和 Getter 方法 
} 


在 find0 方 法 中 含有 4 个 参数 ， 其 中 ，-1 参数 分 别 代表 当前 页 数 和 每 页 显示 的 记录 数 ，where 参数 
代表 的 是 查询 条 件 ，null 参数 代表 的 是 数据 排序 的 条 件 参 数 。find0 方 法 会 根据 提供 的 两 个 -1 参数 执行 
以 下 代码 。 

// 如 果 maxResult<0， 则 查询 所 有 

if(maxResult < 0 && pageNo < 0){ 

list = querylist(); /将 查询 结果 转化 为 List 对 象 

} 


23.8.3 ”商品 搜索 的 实现 


在 Go 购 商 城中 主要 实现 普通 搜索 , 在 对 数据 表 的 简单 搜索 中 , 当 搜 索 表单 中 没有 输入 任何 数据 时 ， 
单 击 “搜索 ”按钮 后 ， 可 以 对 数据 表 中 的 所 有 内 容 进行 查询 ， 当 在 关键 字 文 本 框 中 输入 要 搜索 的 内 容 ， 
单 击 “ 搜 索 ” 按 钮 后 ， 可 以 按 关键 字 内 容 查询 数据 表 中 所 有 的 内 容 。 该 功能 方便 了 用 户 对 商品 信息 的 
查找 ， 用 户 可 以 在 首页 的 文本 输入 框 中 输入 关键 字 搜索 指定 的 商品 信息 ， 如 图 23.17 所 示 。 
MNmeam 刁 网 | 兴 


首页 新 品 上 市 关 出 两 吕 |。 我 的 王 户 车 言 怕 


地 所 阿 品 。 多 件 








品名 际 : :rn 坑 W 
一 市 场 从 





GORMAN: 1190 P800 





Er 
区 区 
sa Ca 
[35 





六 NTTRE 且 和 
市 场 价格 : 358 
cope: 1200 E00 


区 苞 TREE 
图 23.17 商品 搜索 的 效果 


商品 搜索 的 方法 封装 在 ProductAction 类 中 ， 通 过 HQL 的 like 条 件 语句 实现 商品 的 模糊 查询 的 功 
能 ， 其 关键 代码 如 下 。 
代码 位 置 : 光盘 \TM\sl23\Shop\src\com\lyq\action\product\ProductAction 


public String findByName!() throws Exception { 
iproduct.getName() != null{f 
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String where = "where name like ?"; /| 查询 的 条 件 语句 
Objectl queryParams = {"%" + product.getName() + "%"}; /为 参数 赋值 
pageModel = productDao .find(pageNo, pageSize, where, queryParams ); /执行 查询 方法 

} 

return LIST; /1 返回 列表 首页 


于 
在 商品 的 列表 页 面 中 , 通过 Struts 2 的 <s:iterator> 标 签 遍历 返回 的 商品 List 集合, 其 关键 代码 如 下 。 
代码 位 置 ， 光盘 \TM\sl\23\Shop\WebContent\WWEB-INF\pages\product\product listjsp 
<s:iterator value="pageModel.list"> 
<ul> 
<li> 
<table border="0" width="100%" cellpadding="0" cellspacing= 
<tr> 
<td rowspan="5" width="160"> 
<s:a action="product_select" namespace="/product"> 
<s:param name="id" value="id"></s:param> 
<img width="150" height="150"src="<s:property 
value="#request.get(javax.servlet.forward.context_path')"/>/upload 
/<s:property value="uploadFile.path"/>"> 
</s:a></td> 
</tr> 
<tr bgcolor="#f2eec9"> 
<td align="right" width="90"> 商 品名 称 : </td> 
<td><s:a action="product_select" namespace="/product"> 
<s:param name="id" value="id"></s:param> 
<s:property value="name" /> 
</s:a></td> 
</tr> 
<tr> 
<td align="right" width="90"> 市 场 价格 : </td> 
<td><font style="text-decoration: line-through;"> 
<s:property value="marketprice" /> </font></td> 
</tr> 
<tr bgcolor="#f2eec9"> 
<td align="right" width="90">Go 购 价 格 : </td> 
<td><s:property value="sellprice" /> 
<s:if test="sellprice <= marketprice"> 
<font color="red"> 节 省 
<s:property value="marketprice-sellprice" /></font> 
</s:if></td> 
</tr> 
<tr> 








三 














<td colspan="2" align="right"> 
<s:a action="product_select" namespace="/product"> 
<s:param name="id" value="id"></s:param> 
<img src="${context_path}/css/images/gm_06.gif" width="136" 
height="32" /> 
</s:a></td> 
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</tr> 
</table> 
</l> 
</ul> 
</ s:iterator > 


23.8.4 ”前台 商品 其 他 查询 的 实现 


人 气 商 品 推荐 模块 (如 图 23.18 所 示 ) 、 推 荐 商品 模块 (如 图 23.19 所 示 ) 和 热 销 商品 模块 (如 图 23.20 
所 示 ) 3 个 查询 方式 的 实现 方式 基本 相同 ， 都 是 通过 条 件 语 句 进行 排序 查询 ， 只 不 过 查询 的 条 件 不 同 。 











[推荐 商品 ) 扒 销 商品 | 
区 光 区 外 
Tava 编 程 词 由 。 Cs 编程 词 和 编程 词 内 。 C# 编 程 词 和 
| 六 Java 编程 词典 (人气 : 1737 医 [ 受 
NET 编 程 词典 NET 编 程 词典 。 VB 编程 词典 
图 23.18 ”人气 商 品 模块 图 23.19 推荐 商品 模块 图 23.20 ” 热 销 商品 模块 


1. 人 气 商品 模块 的 实现 


人 气 商 品 的 定义 是 按照 商品 的 浏览 量 最 多 进行 定义 的 ， 系 统 将 筛选 出 商品 中 浏览 量 最 多 的 几 件 商 
品 进行 展示 , 商品 结果 集中 的 信息 将 按照 商品 浏览 量 进行 倒序 排列 , 其 实现 的 方法 封装 在 ProductAction 
类 中 ， 其 关键 代码 如 下 。 

代码 位 置 : 光盘 \TM\sl23\Shop\src\com\lyq\action\product\ProductAction 


public String findByClick() throws Exception{ 
Map<String, String> orderby = new HashMap<String, String>(); /定义 Map 集合 


orderby.put("clickcount", "desc"); /为 Map 集合 赋值 
pageModel = productDao.find(1, 8, orderby ): /| 执行 查找 方法 
return "clickList"; /返回 product_click_listjsp 页 面 


} 


在 调用 的 find0 方 法 中 使 用 了 3 个 参数 ， 前 两 个 参数 设置 显示 的 起 始 位 置 和 显示 记录 数 ， 最 后 一 个 
参数 是 商品 信息 排序 的 条 件 ， 程 序 最 终 返回 到 product click listjsp 页 面 ， 代 码 如 下 。 
代码 位 置 : 光盘 \TMN\sl23\Shop\WebContent\WEB-INF\pages\product\product_list.jsp 


<s:set var="context_path" value="#request.get(javax.servlet.forward.context_path')"></s:set > 
<table width="193" height="23" border="0" cellpadding="0" cellspacing="0"> 

<s:iterator value="pageModel.list"> 

<tr> 
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<td width="187" valign="middle"> 
<img src="${context_pathjycss/images/h_32.gif' width="20" height="17" /> 
<s:a action="product_select" namespace="/product"> 
"id" value="id"></s:param> 
<s:property value="name"/> (人 气 : 
<span class="red"><s:property value="clickcount"/></span>) 
</s:a> 
</td> 
</tr> 
</s:iterator> 
</table> 


2. 推荐 商品 和 热 销 商品 模块 的 实现 

推荐 商品 和 热 销 商品 模块 的 实现 与 商品 搜索 模块 的 实现 比较 类 似 ， 都 是 通过 hql 的 排序 语句 实现 的 ， 
推荐 商品 为 商品 推荐 字段 commend 为 true 的 商品 , 并 且 按 商品 销量 的 倒序 排列 , 其 实现 的 关键 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\23\Shop\srcvcomlyqvaction\productProductAction 

public String fndByCommend() throws Exception{ 





Map<String, String> orderby = new HashMap<String, String>(); /定义 Map 集合 
orderby.put("sellCount", "desc"); /为 Map 集合 赋值 
String where = "where commend = ?"; /设置 条 件 语句 
Object0 queryParams = {true}; // 设 置 参 数值 
pageModel = productDao .find(where, queryParams, orderby, pageNo, pageSize); /执行 查询 方法 
return "findList"; // 返 回 推荐 商品 页 面 


热 销 商品 为 销售 量 较 多 的 商品 , 只 需 按照 商品 销量 的 倒序 进行 排列 即 可 , 并 以 分 页 的 方式 取出 前 6 
条 信息 ， 其 实现 的 关键 代码 如 下 。 


代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\product\ProductAction 
public String findBySellCount() throws Exception{ 


Map<String, String> orderby = new HashMap<String, String>(); /定义 Map 集合 
orderby.put("sellCount", "desc"); /为 Map 集合 赋值 
pageModel = productDao.find(1, 6, orderby ); // 执 行 查询 方法 
return "findList"; // 返 回 热 销 商品 页 面 


由 
两 个 模块 在 页 面 中 展示 的 信息 方式 相同 ， 仅 以 推荐 产品 为 例 ， 其 代码 如 下 。 
代码 位 置 ， 光盘 \TMVsl\23\Shop\WebContent\WEB-INF\pages\product\product listjsp 
<s:set var="context_path" 
value="#request.get(javax.servlet.forward.context_path')"></s:set> 
<div style="width: 195px;"> 
<s:iterator value="page Model.list"> 
<div style="float: left; width:45%; text-align: center;"> 
<s:a action="product_select" namespace="/product"> 
<s:param name="id" value="id"></s:param> 
<img width="90" height="90" border="1" src="<s:propert value="%{context_path}"/> 
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/upload/<s:property value="uploadFile.path"/>"> 
<p style="width: 80px;"><s:property value="name"/></p> 
</s:a> 
</div> 
</s:iterator> 
</div> 


在 Strts 2 的 前 台 Action 配置 文件 struts-front.xml 中 ， 配 置 前 台 商 品 管理 模块 的 Action 以 及 视图 
映射 关系 ， 关 键 代码 如 下 。 


代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\struts-front.xml 

<!-- 商品 Action --> 

<package name="shop.product" extends="shop-default" namespace="/product"> 
<action name="product_*" class="productAction" method="{1}"> 
<result name="list">/WEB-INF/pages/product/product_list.jsp</result> 
<result name="select">/WEB-INF/pages/product/product_select.jsp</result> 
<result name="clickList">/WEB-INF/pages/product/product_click_list.jsp</result> 
<result name="findList">/WEB-INF/pages/product/product_find_list.jsp</result> 
</action> 

</package> 


23.9 ”购物 车 模块 设计 


购物 车 是 商务 网 站 中 必 不 可 少 的 功能 , 购物 车 的 设计 很 大 程度 上 会 决定 网 站 是 否 受 到 用 户 的 关注 。 
商务 网 站 中 的 购物 车 会 将 用 户 选 购 的 未 结算 的 商品 保存 一 段 时 间 ， 防 止 错误 操作 或 意外 发 生 时 购物 车 
中 的 商品 丢失 ， 方 便 了 用 户 的 使 用 。 所 以 在 Go 购 商 城中 购物 车 也 是 必 不 可 少 的 一 个 模块 。 





23.9.1 ”模块 概述 


Go 购 商 城 购物 车 实现 的 主要 功能 包括 添加 选 购 的 新 商品 、 自动 更 新 选 购 的 商品 数量 、 清空 购物 车 、 
自动 调整 商品 总 价格 以 及 生成 订单 信息 等 。 本 模块 实现 的 购物 车 的 功能 结构 如 图 23.21 所 示 。 
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图 23.21 购物 车 模块 的 功能 结构 图 
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如 果 用 户 需要 选 购 商品 ， 必 须 登 录 ， 和 否则 用 户 无 法 使 用 购物 车 功能 。 当 用 户 进入 购物 车 后 ， 可 以 
进行 结算 、 清 空 购物 车 以 及 继续 选 购 等 操作 。 当 用 户 进入 结算 操作 后 ， 需 要 填写 订单 信息 ， 并 选择 支 
付 方式 ， 当 用 户 确认 支付 时 系统 会 生成 相应 的 订单 信息 。 其 功能 流程 图 如 图 23.22 所 示 。 
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23.22 ”购物 车 流程 图 


23.9.2 ”购物 车 模块 技术 分 析 


在 开发 时 一 定 要 注意 有 时 加 入 购物 车 中 没有 任何 的 商品 采购 信息 ， 当 用 户 确认 订单 的 时 候 ， 系 统 
同样 会 生成 一 个 消费 金额 为 0.0 元 且 无 任何 订单 条 目的 订单 信息 ， 在 系统 中 该 信息 是 没有 任何 意义 的 ， 
而 且 有 可 能 导致 系统 不 可 预知 的 错误 ， 为 了 避免 这 种 情况 的 发 生 ， 需 要 修改 前 台 订 单 的 保存 方法 ， 即 
OrderAction 类 中 的 save(0 方 法 ， 判 断 购物 车 对 象 是 否 为 空 ， 如 果 为 空 则 返回 错误 信息 的 提示 页 面 ， 不 
进行 任何 的 后 续 操作 。 在 save0 方 法 中 添加 如 下 代码 。 

代码 位 置 ， 光盘 \TM\sl23\Shop\src\com\lyq\action\order\OrderAction 

public String save() throws Exception { 





| /省 略 的 代码 
Set<Orderltem> cart = getCart(); // 获 取 购 物 车 
if(cart.isEmpty()X{ / 削 断 条 目 信 息 是 否 为 空 
return ERROR; // 返 回 订单 信息 错误 提示 页 面 
} 

村 /省 略 的 代码 


| 
并 创建 前 台 订单 错误 的 提示 页 面 order_errorjsp， 当 用 户 误 操作 导致 系统 生成 的 错误 订单 信息 将 不 
会 保存 到 数据 库 中 ， 而 是 跳 转 到 错误 提示 页 面 。 





23.9.3 ”购物 车 基本 功能 的 实现 


购物 车 的 基本 功能 包括 向 购物 车 中 添加 商品 、 清 空 购物 车 以 及 删除 购物 车 中 指定 的 商品 订单 条 目 
信息 3 项 功能 ， 购 物 车 的 功能 是 基于 Session 变量 实现 的 ，Session 充当 了 一 个 临时 信息 存储 平台 ， 当 
Session 失效 后 ， 其 保存 的 购物 车 信息 也 将 全 部 丢失 。 其 效果 图 如 图 23.23 所 示 。 
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我 的 购物 车 
商品 名 称 市 场 从 价格 毕生 
国 ¥12007 
Java 编程 间 奥 P5845E 为 您 节 省 : 站 3007E Ee 


所 清空 购物 车 站】 。 【三 伍 续 搞 选 只 他 商品 。。 | 太 收 银 台 结账 








图 23.23 ”购物 车 内 的 商品 信息 
1. 向 购物 车 添加 商品 


购物 车 的 主要 工作 就 是 保存 用 户 的 商品 购买 信息 ， 当 登录 会 员 浏览 商品 详细 信息 ， 并 单 击 页 面 上 
的 “立即 购买 ” 超 链接 时 ， 系 统 就 会 将 该 商品 放 入 购物 车 内 。 

在 本 系统 中 ， 将 购物 车 的 信息 保存 在 Session 变量 中 ， 其 保存 的 是 商品 的 购买 信息 ， 也 就 是 订单 的 
条 目 信 息 。 所 以 在 向 购物 车 添加 商品 时 ， 首 先 要 获取 商品 ID 进行 判断 ， 如 果 购 物 车 中 存在 相同 的 ID 
值 ， 就 修改 该 商品 的 数量 ， 自 动 加 1; 如 果 购 物 车 中 无 相同 ID 的 值 ， 则 向 购物 车 中 添加 新 的 商品 购买 
信息 。 向 购物 车 添加 商品 信息 的 方法 封装 在 CartAction 类 中 ， 其 关键 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\CartAction 

public String add() throws Exception { 








if(productld != null && productld > OX{ 


Set<Orderltem> cart = getCart(); // 获 取 购 物 车 
/ 标记 添加 的 商品 是 否 是 同一 件 商品 
boolean same = false; // 定 义 same 布尔 变量 
for (Orderltem item : cart) { // 人 遍历 购物 车 中 的 信息 
if(item.getProductld() == productld){ 

/购买 相同 的 商品 ， 更 新 数量 

item.setAmount(item.getAmount() + 1); 

same = true; // 设 置 same 变量 为 true 

上 
} 
// 不 是 同一 件 商品 
if(Isame}{ 
Orderltem item = new Orderltem(); /实例 化 订单 条 目 信息 实体 对 象 
ProductInfo pro = productDao.load(productld); /加 载 商品 对 象 
item.setProductld(pro.getld()); /设置 id 
item.setProductName(pro.getName()); // 设 置 商品 名 称 
item.setProductPrice(pro.getSellprice()); // 设 置 商品 销售 价格 
item.setProductMarketprice(pro.getMarketprice()); // 设 置 商品 市 场 价格 
cart.add(item); // 将 信息 添加 到 购物 车 中 
session.put("cart", cart); // 将 购物 车 保存 在 Session 对 象 中 
} 
return LIST; 


MySQL 从 入 门 到 精通 


程序 运行 结束 够 将 返回 订单 条 目 信息 的 列表 页 面 ， 即 cart_ listjsp， 代 码 如 下 。 


代码 位 置 : 光盘 \TMN\sl23\Shop\WebContent\WEB-INF\pages\cart\cart_ listjsp 
/遍历 Session 对 象 : 通过 Struts 2 的 <s:iterator> 标 签 遍历 Session 对 象 中 存放 的 订单 条 目 信息 
<s:iterator value="#session.cart"> 
<s:set value="%{#sumall +productPrice*amount}" var="sumall" /> 
.…<!-- 省 略 的 布局 代码 --> 
<td width="213" height="30" align="center"> 
<s:property value="productName" /></td> 
<td width="130" align="center"> 
<span style="text-decoration: line-through;"> ¥ 
<s:property value="productMarketprice" /> 元 </span></td> 
<td width="130" align="center">¥ 
<s:property value="productPrice" /> 元 <br> 为 您 节省 : 半 
// 计 算 “ 为 您 节省 ”金额 :其 金额 的 计算 公式 为 (市 场 价格 -销售 价格 ) 
<s:propertyvalue="productMarketprice*amount - productPrice*amount" /> 元 </td> 
<td width="104" align="center" class="red"> 
<s:property value="amount" /></td> 
<td width="111" align="center"><s:a action="cart_delete" namespace="/product"> 
<s:param name="productld" value="productld"></s:param> 
<img src="${context_path}/css/images/zh03_03.gif" width="52" height="23" /> 
</s:a></td> 
…<L-- 省 略 的 布局 代码 --> 
</s:iterator > 


2. 删除 购物 车 中 指定 商品 订单 条 目 信息 


当 用 户 想 删除 购物 车 中 的 某 个 商品 的 订单 条 目 信 息 时 ， 可 以 单 击 信息 后 的 “删除 ” 超 链接 ， 就 会 
自动 清除 该 商品 的 订单 条 目 信息 。 实 现 该 方法 的 关键 代码 如 下 。 
代码 位 置 : 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\CartAction 
public String delete() throws Exception { 
Set<Orderltem> cart = getCart(); /获取 购物 车 
/此 处 使 用 lterator， 否 则 出 现 java.util.ConcurrentModificationException 
lterator<Orderltem> it = cart.iterator(); 
while(it.hasNext()X{ // 使 用 迭代 器 遍历 商品 订单 条 目 信 息 
Orderltem item = it.next(); 
if(item.getProductld() == productldX{ 


让 remove(); // 移 除 商品 订单 条 目 信息 

} 

} 

session.put("cart", cart); // 将 清空 后 的 信息 重新 放 入 Session 中 
return LIST; /返回 购物 车 页 面 
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3. 清空 购物 车 

清空 购物 车 的 实现 较为 简单 ， 由 于 信息 是 暂时 存放 于 Session 对 象 中 ， 所 以 用 户 在 执行 清空 购物 车 
操作 时 ， 直 接 清 空 Session 对 象 即 可 。 当 用 户 单 击 购物 车 页 面 中 的 “清空 购物 车 ” 超 链接 时 ， 系 统 会 向 
服务 器 发 送 一 个 cart_clear.html 的 URL 请 求 ， 该 请 求 执行 的 是 CartAction 类 中 的 clear0 方 法 。 

代码 位 置 : 光盘 \TMN\sl23\Shop\src\com\lyq\action\order\CartAction 


public String clear() throws Exception { 
session .remove("cart ); // 移 除 信息 
return LIST; // 返 回 订单 列表 页 面 


} 


4. 查找 购物 信息 

当 用 户 登录 后 ， 可 以 单 击 首页 顶部 的 “购物 车 ”链接 ， 查 看 自己 的 购物 车 的 相关 信息 。 

当 用 户 单 击 “ 购 物 车 ” 超 链接 后 ， 系 统 会 发 送 一 个 cart_list.html 的 URL 请 求 ， 该 请 求 执行 的 是 
CartAction 中 的 list0 方 法 ， 实 现 该 方法 的 关键 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\CartAction 


public String list() throws Exception { 
return LIST; // 返 回 购物 车 页 面 


} 
在 购物 车 页 面 中 是 通过 Stmts 2 的 <s:iterator> 标 签 遍历 Session 对 象 中 购物 车 的 相关 信息 的 ， 在 程 
序 模块 中 并 不 需要 执行 任何 的 操作 ， 只 需要 返回 购物 车 页 面 即 可 。 
在 Stmuts 2 的 前 台 Action 配置 文件 struts-front.xml 中 ， 配 置 购物 车 管理 模块 的 Action 以 及 视图 映 
射 关系 ， 关 键 代码 如 下 。 
代码 位 置 ， 光 盘 \TM\sl\23\Shop\src\com\lyq\action\struts-front.xml 
<!-- 购物 车 Action --> 
<package name="shop.cart" extends="shop.front" namespace="/product"> 
<action name="cart_*" class="cartAction" method="{1}"> 
<result name="list">/WEB-INF/pages/cart/cart_list.jsp</result> 
<interceptor-ref name="customerDefaultStack"/> 
</action> 
</package> 


23.9.4 订单 相关 功能 的 实现 


要 为 选 购 的 商品 进行 结算 ， 就 需要 先生 成 一 个 订单 ， 订 单 信 息 中 包括 收 货 人 信息 、 送 货 方式 、 支 
付 方 式 、 购 买 的 商品 以 及 订单 总 价格 ， 当 用 户 在 购物 车 中 单 击 “ 收 银 台 结 账 ” 超 链 接 后 ， 将 进入 到 订 
单 填写 的 页 面 ， 其 中 包含 订单 的 基本 信息 ， 例 如 收 货 人 姓名 、 收 货 人 地 址 、 收 货 人 电话 以 及 支付 方式 ， 
该 页 面 为 order_addjsp， 如 图 23.24 所 示 。 下 面 介绍 实现 过 程 。 
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车 杯 下 长 下 市 二 末 医 xxx 号 xxx 小 区 xxx 站 
收据 人 地 


必 红 人 电话 :3xoo6668 


同上 名 和 
雪人 方式 ，“ 支 全 

lf 二 

个 邮 夺 江 坎 收 甘 地址: 计 条 生长 覃 市 st 大 天 nx 室 收 车 人 姓名 : xm 收 甘 他 纺 : sam 





付 一 歌 
23.24 ”Go 购 商 城 订单 信息 添加 页 面 








1. 下 订单 操作 

当 用 户 单 击 “ 收 银 台 结账 ” 超 链接 时 ， 系 统 将 发 送 一 个 order_add.html 的 URL 请 求 ， 该 请 求 执行 
的 是 OrderAction 类 中 的 add0 方 法 ,通过 该 方法 将 用 户 的 基本 信息 从 Session 对 象 中 取出 ， 添 加 到 订单 
表单 中 指定 的 位 置 ， 并 跳 转 到 我 的 订单 页 面 ， 其 关键 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\OrderAction 

public String add() throws Exception { 


order.setName(getLoginCustomer().getUsername!()); // 设 置 收 货 人 姓名 
order.setAddress(getLoginCustomer().getAddress()); /设置 收 货 人 地 址 
order.setMobile(getLoginCustomer().getMobile()); // 设 置 收 货 人 电话 
return ADD; // 返 回 我 的 订单 页 面 
} 
2. 订单 确认 


在 我 的 订单 页 面 单 击 “ 付 款 ” 超 链接 ， 如 图 23.24 所 示 , 将 进入 订单 确认 的 页 面 ， 如 图 23.25 所 示 。 
在 该 页 面 将 显示 订单 的 条 目 信 息 ， 也 就 是 用 户 购买 商品 的 信息 清单 ， 以 便 用 户 进行 确认 。 


我 的 信息 

疏 顷 人 挂名 : mrsof 

疏 估 人 地 址 ; 十 林 省 长 市 二 道 区 xm 号 xm 小 区 zei 门 
疏 线 人 昌 话 :139ccoxtg38 

二 雪 去 付 方式 : 邮局 汇 元 


(a 








图 23.25 订单 确认 页 面 
当 用 户 单 击 我 的 订单 页 面 中 的 “付款 ” 超 链接 时 ， 系 统 将 发 送 一 个 order_confirm.html 的 URL 请 
求 , 该 请 求 执行 的 是 OrderAction 类 中 的 confirm0 方 法 ,该 方法 中 只 是 实现 页 面 的 跳 转 操作 ， 其 关键 代 
码 如 下 。 
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代码 位 置 : 光盘 \TM\sl23\Shop\src\com\lyq\action\order\OrderAction 


public String confirm() throws Exception { 
return "confirm"; 


} 








该 方法 将 返回 order_ confirmjsp， 该 页 面 即 为 订单 确认 页 面 ， 其 订单 条 目 信息 的 显示 与 购物 车 页 面 








中 订单 条 目 信 息 显示 的 方法 相同 。 
3. 订单 保存 


/返回 订单 确认 页 面 





在 订单 确认 页 面 单 击 “ 付 款 ” 超 链接 时 ， 系 统 将 正式 生成 用 户 的 购物 订单 ， 标 志 着 正式 的 交易 开 
始 进行 ， 该 链接 将 会 触发 OrderAction 类 中 的 save0 方 法 ，save0 方 法 将 把 订单 信息 保存 到 数据 库 ， 其 关 


键 代 码 如 下 。 


代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\OrderAction 
public String save() throws Exception { 

if(getLoginCustomer() != null){ 

order.setOrderld(StringUitl.createOrderld()); 

order.setCustomer(getLoginCustomer()); 

Set<Orderltem> cart = getCart(); 

/依次 将 更 新 订单 项 中 的 商品 的 销售 数量 

for(Orderltem item : cart)}{ 

Integer productld = item.getProductld(); 

ProductInfo product = productDao.load(productld); 

product.setSellCount(product.getSellCount() + iem.getAmount()); 

productDao.update(product); 

} 

order.setOrderltems(cart); 
order.setOrderState(OrderState.DELIVERED); 

float totalPrice = Of; 

for (Orderltem orderltem : cart) { 

totalPrice += orderltem.getProductPrice() * orderltem.getAmount(); 


} 
order.setTotalPrice(totalPrice); 
orderDao.save(order); 
session.remove("cart"); 


} 


return findByCustomer(); 


// 如 果 用 户 已 登录 
// 设 置 订单 号 

// 设 置 所 属 用 户 

// 获 取 购 物 车 


// 人 遍历 购物 车 中 的 订单 条 目 信息 
/获取 商品 ID 

/装载 商品 对 象 

// 更 新 商品 销售 数量 

/修改 商品 信息 


/设置 订单 项 

/设置 订单 状态 

// 计 算 总 额 的 变量 

// 人 遍历 购物 车 中 的 订单 条 目 信息 
// 商 品 单价 * 商 品 数量 


// 设 置 订单 的 总 价格 
/保存 订单 信息 
/清空 购物 车 


// 返 回 消费 者 订单 查询 的 方法 


执行 save0 方 法 后 将 返回 订单 查询 的 方法 fndByCustomer0， 在 该 方法 中 将 以 登录 用 户 的 ID 为 查 











询 条 件 ， 查 询 该 用 户 的 所 有 订单 信息 ， 其 关键 代码 如 下 。 


代码 位 置 : 光盘 \TM\sl23\Shop\src\com\lyq\action\order\OrderAction 
public String findByCustomer() throws Exception { 
if(getLoginCustomer() != null}{ 
String where = "where customer.id = ?"; 
Object[] queryParams = {getLoginCustomer().getld()}; 
Map<String, String> orderby = new HashMap<String, String>(1); 
orderby.put("createTime", "desc"); 








// 如 果 用 户 已 登录 

// 将 用 户 id 设置 为 查询 条 件 
1/ 创建 对 象 数组 

// 创 建 Map 集合 
/设置 排序 条 件 及 方式 
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pageModel = orderDao.find(where, queryParams, orderby , pageNo, pageSize);// 执 行 查询 方法 


} 
return LIST; 1/ 返回 订 单列 表 页 面 
出 


该 方法 将 返回 用 户 的 订单 列表 页 面 order listjsp。 
在 Stmuts 2 的 前 台 Action 配置 文件 struts-front.xml 中 ， 配 置 前 台 订 单 管理 模块 的 Action 以 及 视图 
映射 关系 ， 关 键 代码 如 下 。 


代码 位 置 ， 光盘 \TMN\sl23\Shop\src\com\lyq\action\struts-front.xml 

< 上 - 订单 Action --> 

<package name="shop.order" extends="shop.front" namespace="/product"> 
<action name="order_” class="orderAction" method="{1}"> 
<result name="add">/WEB-INF/pages/order/order_add.jsp</result> 
<result name="confirm">/WEB-INF/pages/order/order_confirm.jsp</result> 
<result name="list">/WEB-INF/pages/order/order_list.jsp</result> 
<result name="error">/WEB-INF/pages/order/order_error.jsp</result> 
<interceptor-ref name="customerDefaultStack"/> 
</action> 

</package> 














23.10 ”后 台 商 品 管理 模块 设计 


商品 是 Go 购 商 城 的 灵魂 ， 如 何 管理 好 琳琅 满目 的 商品 信息 也 是 Go 购 商 城 后 台 管 理 的 一 个 难题 ， 
良好 后 台 商 品 管理 机 制 是 一 个 商务 网 站 的 基石 ， 如 果 没 有 商品 信息 维护 ， 商 务 网 站 将 没有 意义 。 


23.10.1 ”模块 概述 


根据 商务 网 站 的 基本 要 求 , Go 购 商 城 网 站 的 商品 管理 模块 主要 实现 商品 信息 查询 、 修 改 商 品 信息 、 
删除 商品 信息 以 及 添加 商品 信息 等 功能 。 后 台 商 品 管理 模块 的 框架 图 如 图 23.26 所 示 。 





后 台 商品 管理 





所 所 商 
商 商 商 有 有 品 
品 品 品 商 商 类 
添 修 删 品 品 别 
加 改 除 查 类 添 


























图 23.26 后 台 商品 管理 模块 框架 图 
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23.10.2 后台 商品 管理 模块 技术 分 析 


解决 Struts2 的 乱码 问题 可 以 在 struts.properties 文件 中 进行 如 下 配置 。 

struts.i18n.encoding=UTF-8 

struts.il8n.encoding 用 来 设置 Web 的 默认 编码 方式 , Go 购 商 城 使 用 了 UTF-8 作为 默认 的 编码 方式 ， 
虽然 该 方法 可 以 有 效 解决 表单 的 中 文 乱码 问题 ， 但 是 该 模式 要 求 表单 的 method 属性 必须 为 post， 由 于 
Struts 2 中 的 form 表单 标签 默认 的 method 属性 就 为 post， 所 以 不 必 再 进行 额外 的 设置 ， 如 果 页 面 中 的 
表单 没有 使 用 Struts 2 的 表单 标签 ， 需 要 在 表单 中 指定 method 的 属性 值 。 





23.10.3 ”商品 管理 功能 的 实现 


在 商品 管理 的 基本 模块 中 ， 包 括 商 品 的 查询 、 修 改 、 删 除 以 及 添加 等 功能 。 
1. 后 台 商 品 查询 


在 Go 购 商 城 的 后 台 管 理 页 面 中 ， 单 击 左 侧 导航 栏 中 的 “查看 所 有 商品 ” 超 链接 ， 显 示 所 有 商品 查 
询 页 面 的 运行 效果 如 图 23.27 所 示 。 





D 商品 名 称 所 属 类 别 。 ”条 购 价格 。 “销售 价格 。 “是 否 推荐 。 ”适应 性 别 。 “编辑 。。 删除 





1 Java 编程 词典 软件 1200 troe 
2 尼 编 程 词典 软件 1200 
NET 编 程 词典 软件 1200 








首页 上 -页 [1] 下 -页 屋 责 
图 23.27 后 台 商 品 信息 列表 页 面 
后 台 商 品 列表 页 面 实现 的 关键 代码 如 下 。 


代码 位 置 : 光盘 \TMN\sl23\Shop\WebContent\WEB-INF\pages\admin\product\ product listjsp 
< table width="693" height="29" border="0" class="word01"> 

<tr> 

<td width="37" height="27" align="center">ID</td> 

<td width="120" align="center"> 商 品名 称 </td> 

<td width="78" align="center"> 所 属 类 别 </td> 

<td width="79" align="center"> 采 购 价格 </td> 

<td width="79" align="center"> 销 售 价格 </td> 

<td width="79" align="center"> 是 否 推荐 </td> 

<td width="79" align="center"> 适 应 性 别 </td> 

<td width="52" align="center"> 编 辑 </td> 

<td width="52" align="center"> 删 除 </td> 
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</tr> 

</table> 

</div> 

<div id="right_mid"> 

<div id="tiao"> 

<table width="693" height="29" border="0"> 
<s:iterator value="page Model.list"> 
<tr> 
<td width="37" height="27" align="center"><s:property value="id" /></td> 
<td width="120" align="center"><s:a action="product_edit" namespace="/admin/product"> 
<s:param name="id" value="id"></s:param><s:property value="name" /></s:a></td> 
<td width="78" align="center"><s:property value="category.name" /></td> 
<td width="79" align="center"><s:property value="baseprice" /></td> 
<td width="79" align="center"><s:property value="sellprice" /></td> 
<td width="79" align="center"><s:property value="commend" /></td> 
<td width="79" align="center"><s:property value="sexrequest.name" /></td> 
<td width="52" align="center"><s:a action="product_edit" namespace="/admin/product"> 
<s:param name="id" value="id"></s:param> 
<img src="${context_path}/css/images/rz_119.gif" width="21" height="16" /></s:a></td> 
<td width="52" align="center"><s:a action="product_del" namespace="/admin/product"> 
<s:param name="id" value="id"></s:param> 
<img src="${context_path}/css/images/rz_17.gif" width="15"height="16" /></s:a></td> 
</tr> 
</s:iterator> 

</table> 


当 用 户 单 击 该 链接 时 系统 将 会 发 送 一 个 product listhtml 的 URL 请 求 ， 该 请 求 执行 的 是 
ProductAction 类 中 的 list0 方 法 ，ProductAction 类 继承 了 BaseAction 类 和 ModelDriven 接口 ， 其 关键 代 
码 如 下 。 

代码 位 置 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\ProductAction 


public String list() throws Exception{ 

pageModel = productDao.find(pageNo, pageSize); // 调 用 公共 的 查询 方法 
return LIST; // 返 回 后 台 商品 列表 页 面 

} 


当 用 户 单 击 列表 中 的 商品 名 称 超 链 接 或 是 列表 中 的 对 图 标 时 , 将 进入 商品 信息 的 编辑 页 面 ,如 图 23.28 
所 示 。 在 该 页 面 可 以 对 商品 的 信息 进行 修改 , 该 操作 触发 的 是 商品 详细 信息 的 查找 方法 , ProductAction 
类 中 的 edit0 方 法 ， 该 方法 将 以 商品 的 ID 值 作为 查询 条 件 ， 其 关键 代码 如 下 。 

代码 位 置 : 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\ProductAction 

public String edit() throws Exception{ 





this.product = productDao.get(product.getld()); /执行 封装 的 查询 方法 
createCategoryTree(); /| 生成 商品 的 类 别 树 
return EDIT; // 返 回 商品 信息 编辑 页 面 
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商品 名 称 : Java 坊 程 词 册 
选择 类 别 : 
采购 价格 : 9 
市 场 价格 : 
销售 价格 : 
是 否 为 推荐 : 
所 属性 别 : 
上 传 图 片 : 








商品 说 明 : 总 











23.28 ”商品 信息 编辑 页 面 


商品 信息 编辑 页 面 与 商品 添加 页 面 的 实现 代码 是 基本 相同 的 ， 区 别 是 在 编辑 页 面 中 需要 显示 查询 
到 的 商品 信息 ， 商 品 编辑 页 面 的 关键 代码 如 下 。 
代码 位 置 ， 光盘 \TM\sl23\Shop\WebContent\WEB-INF\pages\admin\product\product_edit.jsp 
<table width="685" height="24" border="0"> 
<tr> 
<td width="119" height="22" bgcolor="#c6e8ff" align="right"> 商 品名 称 : </td> 
<td width="119" height="22" bgcolor="#c6e8fr" align="right"> 
<s:textfield name="name"></s:textfield></td> 
<td rowspan="7"> 
<img width="270" height="180" border="1" src="<s:property 
value="#request.get('javax.servlet.forward.context_path')"/> 
/upload/<s:property value="uploadFile.path"/>"></td> 
</tr> 
<tr> 
<td width="119" height="22" bgcolor="#c6e8ff' align="right"> 选 择 类 别 : </td> 
<td colspan="2"> 
<s:select name="category.id" list="map" value="category.id"></s:select></td> 
</tr> 
<tr> 
<td width="119" height="22" bgcolor="#c6e8ff' align="right"> 采 购 价格 : </td> 
<td colspan="2"><s:textfield name="baseprice"></s:textfield></td> 
</tr> 
<tr> 
<td width="119" height="22" bgcolor="#c6e8fr" align="right"> 市 场 价 格 : </td> 
<td colspan="2"><s:textfield name="marketprice"></s:textfield></td> 
</tr> 
<tr> 
<td width="119" height="22" bgcolor="#c6e8ff' align="right"> 销 售 价格 : </td> 
<td colspan="2"><s:textfield name="sellprice"></s:textfield></td> 
</tr> 
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<tr> 
<td width="119" height="22" bgcolor="#c6e8ff' align="right"> 是 否 为 推荐 : </td> 
<td colspan="2"> 
<s:radio name="commend" list="#{true":' 是 ','false":' 否 '" value="commend"></s:radio></td> 
</tr> 
<tr> 
<td width="119" height="22" bgcolor="#c6e8ff' align="right"> 所 属性 别 : </td> 
<td colspan="2"> 
<s:select name="sexrequest" list="@com.lyq.model.Sex@getValues()" 
value="sexrequest.getName!()"></s:select></td> 
</tr> 
<tr> 
<td width="119" height="22" bgcolor="#c6e8ff" align="right"> 上 传 图 片 : </td> 
<td colspan="2"><s:file id="file" name="file"></s:file></td> 
</tr> 
<tr> 
<td width="119" height="22" bgcolor="#c6e8ff' align="right"> 商 品 说 明 : </td> 
<td colspan="2"><s:textarea name="description" cols="50" rows="6"></s:textarea></td> 
</tr> 
</table> 


2. 商品 修改 

当 用 户 编辑 完 商品 信息 ， 单 击 页 面 中 的 “提交 ”按钮 ， 系 统 将 会 把 用 户 修改 后 的 信息 保存 到 数据 
库 中 ， 该 操作 会 发 送 一 个 product_save.html 的 URL 请 求 ， 它 会 调用 ProductAction 类 中 的 save0 方 法 ， 
在 save() 方 法 中 包括 图 片 的 上 传 和 向 数据 表 中 添加 数据 的 操作 ， 其 具体 的 实现 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\ProductAction 


public String save() throws Exception{ 
if(file != null X // 如 果 文件 路 径 不 为 空 
1/ 获取 服务 器 的 绝对 路 径 

String path = ServletActionContext.getServletContext().getRealPath("/upload"); 
File dir = new File(path); 


if(Idir.exists()X{ // 如 果 文 件 夹 不 存在 
dir.mkdir(); /创建 文件 夹 
} 
String fileName = StringUiti.getStringTime() + "jpg"; // 自 定义 图 片 名 称 
FileInputStream fis = null; // 输 入流 
FileOutputStream fos = null; /输出 流 
try{ 
fis = new FilelnputStream(file); /根据 上 传 文件 创建 InputStream 实例 
fos = new FileOutputStream(new File(dir,fileName)); /创建 写 入 服务 器 地 址 的 输出 流 对 象 
byte[ bs = new byte[1024 * 4]; /创建 字 节 数组 实例 
int len = -1; 
while((len = fis.read(bs)) (= -1){ /循环 读 取 文件 
fos.write(bs, 0, len); /向 指定 的 文件 夹 中 写 数据 
} 
UploadFile uploadFile = new UploadFile(); /实例 化 对 象 
uploadFile.setPath(fileName); /| 设置 文件 名 称 


478 


第 23 章 Struts 2+Spring+HibematetMySQL 实现 网 络 商城 


product.setUploadFile(uploadFile); /设置 上 传 路 径 
} catch (Exception e){ 

e.printStackTrace(); 

}finally{ 

fos.flush(); 

fos.close(); 

fis.close(); 


} 


} 

// 如 果 商 品类 别 和 商品 类 别 ID 不 为 空 ， 则 保存 商品 类 别 信息 
if(product.getCategory() != null && product.getCategory().getld() != null){ 
product.setCategory(categoryDao.load(product.getCategory().getld())); 


} 

/如 果 上 传 文件 和 上 传 文件 ID 不 为 空 ， 则 保存 文件 的 上 传 路 径 信息 
if(product.getUploadFile() != null && product.getUploadFile().getld() = null)f 
product.setUploadFile(uploadFileDao.load(product.getUploadFile().getld())); 


ee /保存 商品 信息 
return list(); // 返 回 商品 的 查询 方法 

} 

文件 的 上 传 是 网 络 中 应 用 最 为 广泛 的 一 种 技术 ， 在 Web 应 用 中 实现 文件 上 传 需要 通过 Form 表单 
实现 ， 此 时 表单 必须 以 POST 方式 提交 (Struts 2 标签 的 form 表单 默认 提交 方式 为 POST) ， 并 且 必 须 
设置 enctype ="multipart/form-data" 属 性 ， 在 表单 中 需要 实现 一 个 或 多 个 文件 选择 框 供用 户 选择 文件 。 
当 提 交 表 单 后 ， 选 择 的 文件 内 容 会 通过 流 的 方式 进行 传递 ， 在 接收 表单 的 Servlet 或 JSP 页 面 中 获取 该 
流 并 将 流 中 的 数据 读 到 一 个 字 节 数组 中 ， 此 时 字 节 数组 中 存储 了 表单 请 求 中 的 内 容 ， 其 中 包括 所 有 上 
传 文件 的 内 容 ， 因 此 还 需要 从 中 分 离 出 每 个 文件 自己 的 内 容 ， 最 后 将 分 离 出 的 这 些 文件 写 到 磁盘 中 ， 
完成 上 传 操作 。 需 要 注意 的 是 ， 在 进行 分 离 的 过 程 中 ， 操 作 的 内 容 是 以 字 节 形式 存在 的 。 

3. 商品 删除 

当 用 户 单 击 列表 中 的 X 图 标 ， 将 执行 商品 信息 的 删除 操作 ， 该 操作 将 会 向 系统 发 送 一 个 
product_del.html 的 URL 请 求 , 它 将 触发 ProductAction 类 中 的 del0 方 法 , 该 方法 将 以 商品 的 ID 为 参数 ， 
执行 持久 化 类 中 封装 的 delete0 方 法 , delete0 方 法 中 调用 的 是 Hibernate 的 Session 对 象 中 的 delete0 方 法 ， 
其 关键 代码 如 下 。 


代码 位 置 : 光盘 \TM\sl23\Shop\src\com\lyq\action\order\ProductAction 
public String del() throws Exception{ 

















productDao.delete(product getld()): /| 执行 删除 操作 

return list(); // 返 回 商品 列表 查找 方法 
则 
4. 商品 添加 











当 用 户 单 击 后 台 管 理 页 面 左 侧 导 航 栏 中 的 “商品 添加 ” 超 链 接 时 ， 将 会 进入 商品 添加 的 页 面 ， 如 
图 23.29 所 示 。 
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攻 添加 商品 


商品 名 称 : 

选择 类 别 : 服装 ~ 
采购 价格 : 
市 场 价格 : 
销 告 价格 : 
是 否 为 推荐 : 个 
所 属性 别 : 男 
上 传 图 片 : 浏览 .… 





商品 说 明 : 











23.29 ”商品 的 添加 页 面 


用 户 编辑 完 商 品 信息 ， 单 击 页 面 中 的 “提交 ”按钮 ， 该 操作 将 会 向 系统 发 送 一 个 product_save.html 
的 URL 请 求 ， 它 与 商品 修改 触发 的 是 一 个 方法 ， 都 是 ProductAction 类 中 的 save0 方 法 。 
在 Stmuts 2 的 后 台 Action 配置 文件 struts-admin.xml 中 ， 配 置 商品 管理 模块 的 Action 以 及 视图 映射 
关系 ， 关 键 代 码 如 下 。 
代码 位 置 ， 光 盘 \TM\sl\23\Shop\src\com\lyq\action\struts-admin.xml 
<!-- 商品 管理 -> 
<package name="shop.admin.product" namespace="/admin/product" extends="shop.admin"> 
<action name="product_*" method="{1}" class="productAction"> 
<result name="list">/WEB-INF/pages/admin/product/product_list.jsp</result> 
<result name="input">/WEB-INF/pages/admin/product/product_add.jsp</result> 
<result name="edit">/WEB-INF/pages/admin/product/product_edit.jsp</result> 
<interceptor-ref name="adminDefaultStack"/> 
</action> 
</package> 





23.10.4 商品 类 别管 理 功 能 的 实现 


商品 类 别 的 维护 中 主要 包括 商品 类 别 的 查询 、 修 改 、 删 除 以 及 添加 。 
1. 商品 类 别 查询 


商品 类 别 在 后 台中 分 为 两 种 ， 分 别 是 商品 类 别 树 状 
下 拉 框 的 查询 以 及 商品 类 别 列表 信息 的 查询 ， 商 品类 别 
树 状 下 拉 框 的 查询 的 实现 较为 复杂 一 些 ， 通 过 迭 代 的 方 
式 遍 历 所 有 的 节点 。 

在 后 台 的 商品 类 别 查询 中 ， 通 过 树 状 下 拉 框 的 形式 
展现 给 用 户 ， 如 图 23.30 所 示 。 图 23.30 ”商品 添加 页 面 中 的 商品 类 别 树 状 下 拉 框 
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在 进入 商品 页 面 的 edit0 方 法 中 ， 调 上 
码 如 下 。 
代码 位 置 : 光盘 \TM\sl23\Shop\src\com\lyq\action\product\ProductAction 


private void createCategoryTree(){ 

String where = "where level=1"; /查询 一 级 节点 
PageModel<ProductCategory> pageModel = categoryDao.find(-1, -1,where ,null);// 执 行 查询 方法 
List<ProductCategory> allCategorys = pageModel.getList(); 


了 createCategoryTree0 方 法 用 来 创建 商品 类 别 树 ， 其 关键 代 


map = new LinkedHashMap<lnteger, String>(); // 创 建新 的 集合 
for(ProductCategory category : allCategorysjf{ 1/ 遍 历 所 有 的 一 级 节点 
setNodeMap(map,category,false); /将 其 子 节点 添加 到 集合 中 


} 
由 
在 setNodeMap( 方 法 中 ， 首 先 判断 节点 是 否 为 空 ， 如 果 节 点 为 空 则 停止 遍历 ， 程 序 中 根据 获取 的 
节点 级 别 为 类 别名 称 添 加 字符 串 和 空格 , 用 以 生成 渐进 的 树 状 结构 , 将 拼接 后 的 节点 放 入 Map 集合 中 ， 
并 获取 其 子 节点 重新 调用 setNodeMap0 方 法 ， 直 到 遍历 的 节点 为 空 为 止 ， 其 关键 代码 如 下 。 
代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\product\ProductAction 
private void setNodeMap(Map<Integer, String> map,ProductCategory node,boolean flag){ 


if (node == null) { // 如 果 节 点 为 空 

return; // 返 回 空 ， 结 束 程序 运行 
int level = node.getLevel(); /获取 节点 级 别 
StringBuffer sb = new StringBuffer(); // 定 义 字 符 串 对 象 

if (level > 1){ // 如 果 不 是 根 节点 

for (inti= 0; i < level; i++){ 

sb.append(" "); // 添 加 空格 


} 
sb.append(flag ? "上 : "Lb"); 
| 


// 如 果 为 末节 点 添加 " L"， 反 之 添加 " |" 


map.put(node.getld(), sb.append(node.getName()).toString()); /将 节点 添加 的 集合 中 
Set<ProductCategory> children = node.getChildren(); 1/ 获取 其 子 节点 
// 包含 子 类 别 
if(children != null && children.size() > OX{ // 如 果 节 点 不 为 空 
inti = 0; 
/ 遍历 子 类 别 
for (ProductCategory child : children) { 
boolean b = true; 
if(i == children.size()-1X{ /如 果子 节点 长 度 减 1 为 i, 说 明 为 末节 点 
b =false; // 设 置 布尔 常量 为 false 
} 
setNodeMap(map,child,b); /重新 调用 该 方法 
> 
} 
} 
在 商品 添加 页 面 中 ， 通 过 <s:select> 标 签 将 商品 类 别 树 显 示 在 下 拉 框 中 ， 其 关键 代码 如 下 。 
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代码 位 置 : 光盘 \TMN\s123\Shop\WWEB-INF\pages\admin\product\product_addjjsp 
<tr> 
<td width="119" height="22" bgcolor="#c6e8ff' align="right"> 选 择 类 别 : </td> 
<td><s:select list="map" name="category.id"></s:select></td> 
</tr> 
当 用 户 单 击 后 台 管 理 页 面 左 侧 导 航 栏 中 的 “查询 所 有 类 别 ” 超 链接 时 ， 会 向 系统 发 送 一 个 
category_list.html 的 URL 请 求 , 它 将 会 触发 ProductCategoryAction 类 中 的 list0 方 法 ,其 关键 代码 如 下 。 
代码 位 置 : 光盘 \TMN\sl23\Shop\src\com\lyq\action\product\ProductCategoryAction 
public String list() throws Exception{ 

















Object[0 params = null; /对 象 数组 为 空 

String where; /查询 条 件 变量 

if(pid {= null&& pid > OX // 如 果 有 父 节点 

where = "where parent.id =?"; /执行 查询 条 件 

params = new Integer[pid}; // 设 置 参数 值 

Jelse{ 

where = "where parent is null"; /查询 根 节点 

} 

pageModel = categoryDao .find(pageNo,pageSize,where,params); /| 执行 封装 的 查询 方法 
return LIST; 1/ 返回 后 台 类 别 列表 页 面 


} 
该 方法 将 返回 后 台 的 商品 类 别 列表 页 面 ， 如 图 23.31 所 示 。 
添加 子 类 别 。 ”所属 父 类 








首页 上 -页 [1] 下 = 页 尾 页 
图 23.31 后 台 商品 类 别 信息 列表 


2. 商品 类 别 添 加 
单 击 导航 栏 中 “添加 商品 类 别 ” 或 是 商品 类 别 列表 页 面 中 的 “添加 ” 超 链接 时 ， 会 进入 到 商品 
别 的 添加 页 面 ， 如 图 23.32 所 示 。 


添加 商品 类 别 











类 别名 称 : 


图 23.32 商品 类 别 添加 页 面 


在 “类 别名 称 ”文本 框 中 输入 类 别名 称 后 ， 单 击 “ 提 交 ” 按 钮 ， 将 会 触发 ProductCategoryAction 
类 中 的 save0 方 法 ， 在 save0 方 法 中 首先 判断 该 节点 的 父 节 点 参数 是 否 存 在 ， 如 果 存 在 则 先 设 置 其 父 节 
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点 属性 ， 然 后 再 保存 商品 类 别 信息 ， 其 关键 代码 如 下 。 


代码 位 置 : 光盘 \TMN\sl23\Shop\src\com\lyq\action\product\ProductCategoryAction 
public String save() throws Exception{ 


if(pid != null&& pid > 0 /如果 有 父 节点 
category.setParent(categoryDao .load(pid)); /设置 其 父 节点 

} 

categoryDao.saveOrUpdate(category): /添加 类 别 信息 

return list(); // 返 回 类 别 列表 的 查找 方法 


} 


3. 商品 类 别 修改 
当 网 站 管理 员 单 击 商品 类 别 列表 中 的 E 共 按钮 寺 ， 将 进入 商品 类 别 修改 的 页 面 ， 如 图 23.33 所 示 。 








图 23.33 ”商品 类 别 修改 页 面 


修改 商品 类 别 信 息 完毕 后 ， 单 击 页 面 中 的 “提交 ”按钮 ， 其 触发 的 也 是 商品 类 别 添加 中 
ProductCategoryAction 类 的 save0 方 法 。 

4. 商品 类 别 删除 

当 用 户 单 击 商品 类 别 列表 中 的 兴 图 标 ， 将 执行 商品 类 别 信息 的 删除 操作 ， 该 操作 将 会 向 系统 发 送 
一 个 category_delhtml 的 URL 请 求 ， 它 将 触发 ProductCategoryAction 类 中 的 del0 方 法 ， 该 方法 将 以 商 
品类 别 的 ID 为 参数 ， 执 行 持久 化 类 中 封装 的 delete0 方 法 ， 删 除 指定 的 信息 ， 其 关键 代码 如 下 。 

代码 位 置 : 光盘 \TM\sl23\Shop\src\com\lyq\action\product\ProductCategoryAction 

public String del() throws Exception{ 


if(category.getld() != null && category.getld() > OX{ // 判 断 是 否 获得 ID 参数 
categoryDao.delete(category.getld()); /执行 删除 操作 
return list(); // 返 回 商品 类 别 列表 的 查找 方法 
} 
在 商品 类 别管 理 中 添加 、 修 改 以 及 删除 的 操作 实现 都 较为 简单 ， 商 品类 别 信息 的 查询 方法 支持 无 
限 级 的 树 状 分 级 查询 。 


在 Struts 2 的 后 台 Action 配置 文件 struts-admin.xml 中 ， 配 置 商品 类 别管 理 模块 的 Action 以 及 视图 
映射 关系 ， 关 键 代码 如 下 。 


代码 位 置 : 光盘 \TMVsl\23\Shop\srcvcomlyqvaction\struts-admin .xml 

<!-- 类 别管 理 一 > 

<package name="shop.admin.category" namespace="/admin/product" extends="shop.admin"> 
<action name="category_*" method="{1}" class="productCategoryAction"> 
<result name="list">/WEB-INF/pages/admin/product/category_list.jsp</result> 
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<result name="input">/WEB-INF/pages/admin/product/category_add.jsp</result> 
<result name="edit">/WEB-INF/pages/admin/product/category_edit.jsp</result> 
<interceptor-ref name="adminDefaultStack"/> 
</action> 
</package> 


23.11 后 台 订 单 管理 模块 的 设计 


网 站 管理 员 可 以 对 会 员 的 订单 进行 维护 ， 但 这 种 维护 只 能 修改 订单 的 状态 ， 并 不 能 修改 订单 中 的 
任何 信息 ， 因 为 当 用 户 确认 订单 时 该 订单 就 已 经 生效 ， 它 相当 于 用 户 与 商家 交易 的 一 个 契约 ， 是 用 户 
与 商家 之 间 的 一 个 交易 凭证 ， 所 以 不 能 进行 任何 的 修改 。 


23.11.1 ”模块 概述 





在 后 台 的 订单 管理 模块 中 ， 主 要 分 为 两 个 基本 模块 ， 分 别 是 订单 的 查询 和 订单 状态 的 修改 ， 其 中 
订单 的 查询 又 可 分 为 订单 的 全 部 查询 和 用 户 自 定义 的 条 件 查询 ， 框 架 模块 图 如 图 23.34 所 示 。 
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图 23.34 订单 管理 模块 框架 图 


23.11.2 ”后台 订单 管理 模块 技术 分 析 

















当 用 户 在 订单 列表 页 面 中 ， 单 击 “ 更 新 订单 状态 ”按钮 时 ， 将 会 弹出 提示 对 话 框 ， 让 用 户 选择 修 
改 的 状态 信息 ， 在 订单 列表 页 面 中 通过 模 态 窗 体 的 形式 弹出 该 对 话 框 ， 为 “更 新 ”按钮 绑 定 触发 事件 
的 关键 代码 如 下 。 


代码 位 置 :光盘 \TM\sl\23\Shop\WebContent\pages\admin\order\order listjsp 
<td width="150" align="center"> 
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<s:url action="order_select" namespace="/admin/product" var="order_select"> 

<s:param name="orderld" value="orderld"></s:param></s:url> 
<input type="button" value=" 更 新 订单 状态 "onclick="openWindow('${order_select},350,150):"> 
</td> 


在 弹出 的 子 窗 体 中 ， 如 果 是 模 态 的 ， 子 窗 体 不 关闭 将 无 法 进行 主 窗 体 的 任何 操作 ; 如 果 是 非 模 态 
的 ， 子 窗 体 不 关闭 同样 可 以 进行 主 窗 体 的 操作 。 

根据 页 面 中 的 代码 可 知 ，Action 请 求 order_select 跳 转 的 页 面 为 order_selectjsp， 也 就 是 弹出 的 模 
态 窗 体 。 

当 用 户 在 弹出 的 模 态 窗 体 中 单 击 该 页 面 ， 将 会 向 系统 发 送 一 个 order_update.html 的 URL 请 求 ， 该 
请 求 触发 的 是 OrderAction 类 中 的 update0 方 法 ， 其 关键 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\OrderAction 

public String update() throws Exception { 








OrderState orderState = order.getOrderState(); 1/ 获取 设置 的 订单 状态 

order = orderDao.load(order.getOrderld()); /装载 订单 对 象 
order.setOrderState(orderState); /| 设置 的 订单 状态 
orderDao.update(order); /修改 订单 状态 

return "update"; // 放 回 订单 状态 修改 成 功 页 面 


} 

当 订 单 状 态 修改 成 功 后 , 会 弹出 订单 信息 修改 成 功 的 提示 对 话 框 , 通过 JavaScript 对 该 窗 体 进行 设 
置 ， 该 窗 体 3 秒 后 自动 关闭 ， 并 刷新 主页 面 。 

在 订单 更 新 成 功 的 页 面 中 ， 设 置 窗 体 自动 关闭 的 JavaScript 关键 代码 如 下 。 

代码 位 置 ， 光盘 \TM\sl23\Shop\WebContent\pages\admin\order\ order_update_success.jsp 

<script type="text/javascript"> 


function closewindow(){ 
if(window.openerj{ 
window.opener.location.reload(true); /刷新 父 窗 体 
window.close(); /| 关闭 提示 窗 体 
} 
function clock(){ 
i=i-1; 
ifi > OX // 如 果 i 大 于 0 
setTimeout("clock();",1000); /11 秒 后 重新 调用 clock() 方 法 
Jelse{ 
closewindow(); // 调 用 关闭 窗 体 方法 
} 
上 
vari= 3; // 设 置 ij 值 
clock(); // 页 面 加 载 后 自动 调用 clock() 
</script> 


在 上 述 程序 中 通过 变量 i 来 设置 窗 体 自动 关闭 的 时 间 , 在 clock0 方 法 中 ， 当 i 的 值 为 零 时 调用 关闭 
窗 体 的 方法 ， 并 且 通 过 setTimeout0 方 法 设置 方法 调用 时 间 ， 参 数 “1000” 的 单位 为 毫秒 。 
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23.11.3 后台 订单 查询 的 实现 


在 后 台 订 单 的 查询 中 ， 分 为 所 有 订单 查询 和 订单 的 自 定义 条 件 查 询 ， 在 订单 的 自 定义 查询 中 用 户 

可 以 根据 设 定 不 同 的 查询 条 件 查 询 订 单 的 指定 信息 。 在 本 应 用 程序 中 两 者 调用 的 都 是 同一 个 查询 方法 。 

在 管理 页 面 左 侧 导 航 栏 中 单 击 “ 查 看 订单 ” 超 链接 时 , 将 进入 订单 状态 管理 页 面 , 如 图 23.35 所 示 。 
订单 号 总 金额 。” 消费 者 。 支付 方式 创建 时 间 订单 捧 态 个 改 























201005041012220323561 。 1200 ”mrsoft 邮局 汇款 。 2010 年 05 月 和 日 10:12 已 发 货 更 新 订单 状态 
201004271843190764180 。 240.0 mrsof ”邮局 汇款 。 2010 年 04 月 2 日 18:43 。 已 发 货 更 新 订单 状态 


201004260941250469034 1200 mrsoft 邮局 汇款 。 2010 年 04 月 26 日 09:41 已 发 货 更 新 订单 状态 





首页 上 -页 [1] 下 = 页 尾 页 
图 23.35 订单 状态 管理 页 面 


当 用 户 单 击 左 侧 导 航 栏 中 的 “订单 查询 ” 超 链接 时 , 将 会 进入 订单 查询 条 件 的 自 定义 页 面 , 如 图 23.36 
所 示 。 





加 订单 查询 





订单 号 : 
订单 拓 态 ;: 6 已 发 全 个 交易 完成 已 取消 
会 员 名 : 
收 车 人 : 





图 23.36 订单 查询 条 件 的 自 定义 页 面 


当 用 户 单 击 导航 栏 中 的 “查看 订单 ” 超 链接 或 单 击 订单 查询 条 件 的 自 定义 页 面 中 的 “提交 ”按钮 
时 ， 都 会 向 系统 发 送 一 个 order_list.html 的 URL 请 求 ， 该 请 求 触发 的 是 OrderAction 中 的 list0 方 法 ， 其 
关键 代码 如 下 。 


代码 位 置 ， 光 盘 \TM\sl\23\Shop\src\com\lyq\action\order\OrderAction 

public String list() throws Exception { 
Map<String, String> orderby = new HashMap<String, String>(1);  // 定 义 Map 集合 
orderby.put("createTime", "desc"); /设置 按 创建 时 间 倒序 排列 
StringBuffer whereBuffer = new StringBuffer(™); /创建 字符 串 对 象 
List<Object> params = new ArrayList<Object>(); 
if(order.getOrderld() != null && order.getOrderld().length() > 0){/ 如 果 订 单 号 不 为 空 





whereBuffer.append("orderld = ?"); /以 订单 号 为 查询 条 件 
params.add(order.getOrderld()); // 设 置 参 数 

} 

if(order.getOrderState() = nullX{ // 如 果 订 单 状 态 不 为 空 
if(params.size() > 0) whereBufferappend(" and "); /增加 查询 条 件 
whereBuffer.append("orderState = ?"); // 设 置 订单 状态 为 查询 条 件 


入 
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params.add(order.getOrderState()); /设置 参数 

} 

if(order.getCustomer() = null && order.getCustomer().getUsername() != null 

&& order getCustomer().getUsername().length() > OX{ /如果 会 员 名 不 为 空 
if(params.size() > 0) whereBuffer append(" and "); /增加 查询 条 件 
whereBuffer.append("customer.username = ?"); // 设 置 会 员 名 为 查询 条 件 
params.add(order.getCustomer().getUsername()); // 设 置 参 数 

} 

if(order.getName() != null && order.getName().length()>0X{ // 如 果 收 丈 人 姓名 不 为 空 
if(params.size() > 0) whereBuffer append(" and "); /增加 查询 条 件 
whereBuffer.append("name = ?"); // 设 置 收 款 人 姓名 为 查询 条 件 
params.add(order.getName()); /设置 参数 


/如 果 whereBuffer 为 空 则 查询 条 件 为 空 ， 否 则 以 whereBuffer 为 查询 条 件 

String where = whereBurffer.length()>0 ? "where "+whereBuffer.toString() : ™"; 

pageModel = orderDao.find(where, params .toArray(), orderby, pageNo, pageSize); // 执 行 查询 方法 

return LIST; // 返 回 后 台 订单 列表 
出 


“查看 订单 ” 超 链接 并 没有 为 list0 方 法 传递 任何 的 参数 ， 所 以 最 后 传 给 find0 方 法 的 where 查询 条 
件 字符 串 为 空 ，find( 方 法 将 会 从 数据 库 中 查询 所 有 的 订单 信息 并 按 创 建 时 间 的 倒序 输出 。 

list0 方 法 将 会 返回 后 台 的 订单 信息 列表 页 面 ， 在 该 页 面 利用 Struts 2 的 <s:iterator> 方 法 遍历 输出 返 
回 结果 集中 的 信息 即 可 。 后 台 订 单 信息 列表 页 面 的 关键 代码 如 下 。 


代码 位 置 ， 光 盘 \TM\s123\Shop\WebContent\pages\admin\order\order list.jsp 
<table width="693" height="29" border="0" class="word01"> 
<tr> 
<td width="140" align="center"> 订 单 号 </td> 
<td width="60" align="center"> 总 金额 </td> 
<td width="63" align="center"> 消 费 者 </td> 
<td width="70" align="center"> 支 付 方式 </td> 
<td width="140" align="center"> 创 建 时 间 </td> 
<td width="70" align="center"> 订 单 状态 </td> 
<td width="150" align="center"> 修 改 </td> 
</tr> 
</table > 
<table width="693" height="29" border="0"> 
<s:iterator value="pageModel.list"> 
<tr> 
<td width="140" align="center"><s:property value="orderld" /></td> 
<td width="60" align="center"><s:property value="totalPrice" /></td> 
<td width="63" align="center"><s:property value="customer.username" /></td> 
<td width="70" align="center"><s:property value="paymentWay.getName()" /></td> 
<td width="140" align="center"> 
/人 遍历 操作 
<s:date name="createTime" format="yyyy 年 MM 月 d 日 HH:mm"/></td> 
<td width="70" align="center"><s:property value="orderState.getName()" /></td> 
<td width="150" align="center"> 
<s:url action="order_select" namespace="/admin/product" ar="order_select"> 
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<s:param name="orderld" value="orderld"></s:param> 
</s:url> <input type="button" value=" 更 新 订单 状态 " 
onclick="openWindow!('${order_select},350,150):"></td> 
</tr> 
</s:iterator> 
</table> 


23.12 ”开发 技巧 与 难点 分 析 


23.12.1 解决 订单 号 为 空 时 查询 报错 


在 代码 初期 对 后 台 订 单 自 定义 查询 进行 测试 的 时 候 发 现 ， 如 果 订 单 号 为 空 ， 填 写 后 面 的 查询 条 件 
进行 查询 的 时 候 就 会 报错 ， 如 图 23.37 所 示 。 


Struts Problem Report 


Struts has detected an unhandled exception: 


. unexpected taken: and near line 1，colum 39 [from 
com. 1]yq. model. order. Order where and orderState = ? order by 
createTime desc 

unexpected token: and near line 1，colum 39 [from 
com. lyq. model. order. Order where and orderState = ? order by 
createTime desc]; nested exception is 
org.hibernate. hql. ast. QuerySyntaxException: unexpected token: and| 
near line 1, colum 39 [from com. lyq. model. order.Order where and 
orderState = ? order by createTime desc] 


图 23.37 订单 号 为 空 查 询 时 系统 的 报错 信息 
从 错误 信息 中 可 以 发 现在 where 条 件 查 询 语句 中 关键 字 where 直接 连接 的 是 关键 字 and, 也 就 是 说 
在 订单 状态 前 置 查询 条 件 订 单 号 码 为 空 的 情况 下 同样 将 关键 字 and 拼接 到 了 查询 条 件 中 ， 解 决 该 问题 
只 需 对 查询 条 件 的 Object 数组 的 长 度 进行 判断 ， 如 果 数 组 的 长 度 不 为 0， 就 将 关键 字 and 添加 到 查询 
条 件 中 ， 以 订单 状态 的 查询 条 件 为 例 ， 其 关键 代码 如 下 。 
代码 位 置 ， 光盘 \TM\sl\23\Shop\src\com\lyq\action\order\OrderAction 





if(order.getOrderState() != nullf /1 如果 订单 状态 不 为 空 
if(params.size() > 0){ 
whereBuffer.append(" and "); // 增 加 查询 条 件 
} 
5 /省略 的 代码 
} 


23.12.2 ”通过 Struts 2 的 拦截 器 来 解决 Session 超时 出 现 空 指 针 异 常 的 问题 


从 网 站 的 安全 性 考虑 ， 用 户 在 没有 登录 或 是 Session 失效 的 时 候 是 不 允许 进行 购物 和 后 台 维护 的 ， 
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一 般 客户 端的 Session 是 有 时 间 限 制 的 (根据 服务 器 中 的 配置 决定 ， 一 般 为 20 分 钟 ) ， 如 果 超 出 时 间 
限制 ， 系 统 就 会 报 出 现 空 指针 的 异常 信息 ， 出 现 这 种 情况 的 原因 是 系统 从 Session 中 取得 的 信息 为 空 ， 








即 获取 的 用 户 登 录 信 息 为 空 ， 空 值 造 成 了 这 种 情况 的 发 生 。 





这 个 问题 可 以 通过 Struts 2 的 拦截 器 来 解决 ， 根 据 拦截 器 判断 Session 是 否 为 室 ， 并 根据 判断 结果 








执行 不 同 的 操作 。 


拦截 器 (Interceptor) 是 Struts 2 框架 中 一 个 非常 重要 的 核心 对 象 ， 它 可 以 动态 增强 Action 对 象 的 
功能 ， 通 过 对 登录 拦截 器 的 配置 ， 如 果 会 员 的 Session 失效 后 ， 用 户 将 无 法 使 用 购物 车 功能 ， 除 非 重 新 
登录 ; 如 果 管 理 员 Session 失效 后 ， 将 无 法 进入 后 台 进 行 操作 ， 没 有 直接 登录 的 用 户 在 地 址 栏 中 直接 输 


入 URL 地 址 也 将 被 拦截 器 拦截 ， 并 返回 系统 的 登录 页 面 ， 这 样 很 大 程度 地 提升 了 系统 的 安全 性 。 








拦截 器 动态 地 作用 于 Action 与 Result 之 间 , 它 可 以 动态 地 对 Action 以 及 Result 进行 增强 (在 Action 
与 Result 中 加 入 新 功能 ) 。 当 客户 端 发 送 请 求 时 ， 会 被 Struts 2 的 过 滤器 所 拦截 ， 此 时 Struts 2 对 请 求 
持 有 控制 权 ，Struts 2 会 创建 Action 的 代理 对 象 ， 并 通过 一 系列 的 拦截 器 对 请 求 进行 处 理 ， 最 后 再 交 给 
指定 的 Action 进行 处 理 。 拦 截 器 实现 的 核心 思想 是 AOP(Aspect Oriented Progamming, 面向 切面 编程 ) 。 


首先 创建 会 员 登 录 拦 截 器 CustomerLoginInteceptor， 其 关键 代码 如 下 。 
public String intercept(ActionInvocation invocation) throws Exception { 


ActionContext context = invocation.getlnvocationContext(); // 获 取 ActionContext 
Map<String, Object> session = context.getSession(); 1/ 获取 Map 类 型 的 session 
if(session.get("customer") != null}{ // 判 断 用 户 是 否 登 录 

return invocation.invoke(); // 调 用 执行 方法 

} 

return BaseAction.CUSTOMER_LOGIN; /| 返回 登录 


在 前 台 的 Struts 2 的 配置 文件 中 配置 该 拦截 器 ， 其 关键 代码 如 下 。 


<package name="shop.front" extends="shop-default"> 
<!-- 配置 拦截 器 --> 
<interceptors> 
<!-- 验证 用 户 登录 的 拦截 器 --> 
<interceptor name="|loginInterceptor" class="com.lyq.action.interceptor.CustomerLogininteceptor"/> 
<interceptor-stack name="customerDefaultStack"> 
<interceptor-ref name="loginInterceptor"/> 
<interceptor-ref name="defaultStack"/> 
</interceptor-stack> 
</interceptors> 
<action name="index" class="indexAction"> 
<result>/WEB-INF/pages/index.jsp</result> 
</action> 
</package> 


通常 情况 下 ， 拦 截 器 对 象 实现 的 功能 比较 单一 ， 它 类 似 于 Action 对 象 的 一 个 插件 ， 为 Action 对 象 


动态 地 织 入 新 的 功能 。 
系统 后 台 拦 截 器 的 配置 与 前 台 类 似 ， 这 里 不 再 进行 详细 的 说 明 。 


MySQL 从 入 门 到 精通 


23:13 小 结 


本 系统 只 是 实现 了 电子 商务 网 站 一 些 基本 的 功能 ， 真 正 的 商务 网 站 的 开发 难度 和 工作 量 要 比 本 系 
统 复杂 和 人 烦琐 得 多 ， 但 是 希望 通过 本 系统 的 开发 ， 读 者 可 以 了 解 网 站 开发 的 简单 流程 、SSH2 框架 的 整 
合 以 及 MVC 的 设计 模式 ， 相 信 读 者 可 以 融会 贯通 。 
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